package Debug::ShowStuff;
use strict;
use Carp;
use Tie::IxHash;
use String::Util ':all';
require 5.005; # require at least Perl version 5.005
# version
our $VERSION = '1.16';
# allow for forced context
# This is a change to how Debug::ShowStuff used to work. It used to be that
# the output of a function (e.g. println) was returned as a scalar if the
# function call was made in an array context. This got confusing when the
# call was the last call in a function. I'm changing it so that, by default,
# the function always outputs to the standard file handle (usually STDOUT).
our $always_void = 1;
# allow for forced web/plain
use vars qw[ $forceweb ];
# global output file handle
# default $out to STDOUT
our $out;
$out = *STDOUT;
# default $out to STDOUT
use vars qw[ $active ];
$active = 1;
# INDENT: how far to indent a line
our $indent_level = 0;
my $indent_tab = ' ';
# constants
use constant STDTABLE => qq|
\n
\n|;
#---------------------------------------------------------------------
# export
#
use vars qw[@EXPORT_OK %EXPORT_TAGS @ISA];
use Exporter;
push @ISA, 'Exporter';
@EXPORT_OK = qw[
showhash showhashhtml showhashplain
showarray showarr
showarraydiv showarrdiv
showcgi
showscalar showsca
showref
showstderr
setoutput
pressenter
inweb
httpheader
httpheaders
println
printnorm
printhr
preln
dieln
devexit
diearr
fixundef
diearr
findininc
confirm
output_to_file
define_show
showtainted
showstuff
indent
timer
backtrace
dietrace
autoshow
forceenv
forcetext
forceweb
forcenone
showisa
showsth
showsql
explainsql
];
%EXPORT_TAGS = ('all' => [@EXPORT_OK]);
#
# export
#---------------------------------------------------------------------
#------------------------------------------------------------------------------
# opening POD
#
=head1 NAME
Debug::ShowStuff - A collection of handy debugging routines for displaying
the values of variables with a minimum of coding.
=head1 SYNOPSIS
Here's a sampling of a few of my favorite functions in this module.
use Debug::ShowStuff ':all';
# display values of a hash or hash reference
showhash %hash;
showhash $hashref;
# display values of an array or array reference
showarr @arr;
showarr $arrref;
# show all the params received through CGI
showcgi();
# A particularly fancy utility: display STDERR at top of web page
my $warnings = showstderr;
=head1 INSTALLATION
C can be installed with the usual routine:
perl Makefile.PL
make
make test
make install
=head1 DESCRIPTION
C grew dynamically from my needs in debugging code. I found
myself doing the same tasks over and over... displaying the keys and
values in a hash, displaying the elements in an array, displaying the output
of STDERR in a web page, etc. C began as two or three of my
favorite routines and grew as I added to that collection. Finally I decided
to publish these tools in the hope that other Perl programmers will find
them useful.
=head2 Not for production work
C is for debugging, not for production work. It does not
always output the actual value of something, but rather information B
the value. For example, the following code outputs the actual value in the
first line, but a note about the value in the second.
println 'my value';
println undef;
which outputs
my value
[undef]
I would discourage you from using C in production code.
C is only for quick-n-dirty displays of variable values in
order to debug your code.
=head2 Text and web modes
The functions in C are designed to output either in plain
text mode (like if you're running the script from a command prompt), or in web
mode (like from a CGI). If the script appears to be running in a CGI or other
web mode (see the C function) then values are output using HTML, with
special HTML characters escaped for proper display. Othewise the values are
output as they are.
Generally you won't need to bother telling C if you're in
text or web mode... it figures it out on its own.
=head2 Dynamic output/return: different than previous versions
NOTE: The dynamic behavior of "show" functions has changed since earlier
versions of Debug::ShowStuff. "show" functions now always outputs to STDOUT
or STDERR unless $Debug::ShowStuff::always_void is set to false. By default
$always_void is true.
If $always_void is false, then the following applies:
The functions that start with "show" dynamically either output to STDOUT or
STDERR or return a string, depending on the context in which the
functions are called. For example, if you call showhash in a void context:
showhash %myhash;
then the contents of %myhash are output to STDOUT. On the other hand, if the
function is called in scalar context:
my $var = showhash(%myhash);
then the same string that would have been output to STDOUT is instead
returned and stored in $var.
By default, output is sent to STDOUT, not STDERR. You can change the
default output to STDERR using the C command. See the docs
for that command for more detail.
=head2 Displaying "invisible" strings
To facilitate telling the difference between C<[undef]> and an empty string,
functions output the strings "[undef]" and "[empty string]". So, for example,
this code:
println undef;
println "";
produces this:
[undef]
[empty string]
=head1 FUNCTION DESCRIPTIONS
=cut
#
# opening POD
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# default
# private subroutine
#
sub default {
for (my $i=0; $i<=$#_; $i++) {
defined($_[$i]) and return $_[$i];
}
return undef;
}
#
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# println
#
=head2 println
C was the original Debug::ShowStuff function. It simply outputs the
given values.
In text mode it adds a newline to the end.
For example, this code:
println "hello world";
produces this output, including the newline:
hello world
In L it puts the output inside a element. The values are HTML
escaped so that HTML-significant characters like < and > are actually
displayed as < and >. The
element has CSS styles set so that the
characters are always black, the background is always white, text is
left-aligned, and the
element has a black border, regardless of the styles
of the surrounding elements. So, for example, in web mode, the following code:
println 'whatever';
outputs the following HTML:
whatever
Like other "show" functions, undef is output as the string "[undef]" and
an empty string is output as the string "[empty string]".
Values in the arguments array are output concatenated together with no
spaces in between. Each value is evaluated independently for if it
is undef, empty string, or a string with visible content. So, for example,
this code:
println "whatever", "", undef, "dude";
outputs this:
whatever[empty string][undef]dude
=cut
# And after all that documentation, the sub is only one line long. println
# actually calls the private sub "printer" with a few extra arguments. println
# shares printer with printnorm.
sub println {
return printer (wantarray(), 'p', "\n", @_);
}
#
# println
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# indent
#
=head2 indent()
C is for situations where you're outputting a lot of stuff and you
want to tidy up the putput with indentation.
In L the output is indented with 3 spaces per
indentation level. In web mode the output is indented with 20px per
indentation level.
C must be assigned to a variable or it has no effect. C
increases the indent level by one. When the variable goes out of scope, the
indent level is decreased by one.
For example suppose you want to display the values of records from a database.
You might loop through the records, outputting them like this:
while (my $record = $sth->fetchrow_hashref) {
println $record->{'name_nick'};
my $indent = indent();
showhash $record;
}
That would produce output something like this:
Ricky
---------------------------------------
name_first = Rick
name_last = Adams
---------------------------------------
Dan
---------------------------------------
name_first = Daniel
name_last = Bradley
---------------------------------------
By default, three spaces are used to indent. To change that set
$Debug::ShowStuff::indent_tab to whatever string you want to use for
indentation.
B bottom_space
The C option indicates to output an extra line at the bottom
of the indented output, just to give some extra division before the next
batch of code. For example, the following code does not use C:
foreach my $name (qw[Larry Moe]) {
println $name;
my $indent = indent(bottom_space=>1);
println 'years: ', length($name);
}
and so produces the following output:
Larry
years: 5
Moe
years: 3
But this code:
foreach my $name (qw[Larry Moe]) {
println $name;
my $indent = indent(bottom_space=>1);
println 'years: ', length($name);
}
produces this output:
Larry
years: 5
Moe
years: 3
=cut
sub indent {
my (%opts) = @_;
return Debug::ShowStuff::Indent->new(%opts);
}
#
# indent
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showstuff
#
=head2 showstuff()
This function turns on/off most of the functions in this module, with one
important exception explained below. The function also returns the state of
whether or not Debug::ShowStuff is on/off.
If a parameter is sent, that param is used to turn display on/off. The
value is stored in the global variable $Debug::ShowStuff::active.
The function is also used by most subroutines to determine if they should
actually output anything. $Debug::ShowStuff::active is not the only criteria
used to determine if Debug::ShowStuff is active. The algorithm is as follows:
- If the environment variable $ENV{'SHOWSTUFF'} is defined and false then
showstuff() returns false regardless of the state of $active.
- If the environment variable $ENV{'SHOWSTUFF'} is not defined or is defined
and true then showstuff() uses $Debug::ShowStuff::active to determine on/off.
The purpose of this algorithm is to allow the use of debugging display in
situations where one perl script calls another, such as in regression testing.
For example, suppose you have a script as follows:
#!/usr/bin/perl -w
use strict;
use Debug::ShowStuff ':all';
my ($rv);
println 'running my_function()';
$rv = my_function();
println 'the returned value is: ', $rv;
$rv or die 'error!';
The output of the script might look something like this:
running my_function()
1
Now suppose you call that and other scripts from some OTHER script, and
you don't want the screen cluttered with all that debugging, you just want
to see if all those scripts run successfully. You could use $ENV{'SHOWSTUFF'}
to turn off showing stuff:
#!/usr/bin/perl -w
use strict;
use Debug::ShowStuff ':all';
my @tests = ("./script1.pl", "./script2.pl", "./script3.pl");
$ENV{'SHOWSTUFF'} = 0;
foreach my $test () {
system($test) and die "$test failed";
}
In that case, none of the stuff from the test scripts would be output.
=cut
sub showstuff {
if (@_)
{ $active = $_[0] }
# if SHOWSTUFF environment variable is defined and false
if ( defined($ENV{'SHOWSTUFF'}) && (! $ENV{'SHOWSTUFF'}) )
{ return 0 }
# else use $active
return $active;
}
#
# showstuff
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# htmlesc
#
# Private sub. Formats a string for literal output in HTML. An undefined
# first argument is returned as an empty string.
#
# sub htmlesc {
# my ($rv) = @_;
# return '' unless defined($rv);
# $rv =~ s|&|&|g;
# $rv =~ s|"|"|g;
# $rv =~ s|'|'|g;
# $rv =~ s|<|<|g;
# $rv =~ s|>|>|g;
# return $rv;
# }
#
# htmlesc
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# printnorm
#
=head2 printnorm
Works like println but doesn't add trailing newline. In web environment uses
instead of .
=cut
sub printnorm {
return printer (wantarray(), 'span', '', @_);
}
#
# printnorm
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# printer
# Private sub: called by both println and printnorm
#
sub printer {
my ($context, $html_el, $newline, @args) = @_;
showstuff() or return '';
my ($str);
my $fh = getfh($context);
# special case: no arguments: just output an eol and return
if (! @args) {
print $fh "\n";
return;
}
# get final string, including stuff like [undef]
$str = join('', map {define_show($_)} @args);
# print as web page
if (inweb()) {
my $indent = web_indent();
print $fh
qq|<$html_el style="${indent}background-color:white;color:black;text-align:left">|,
htmlesc($str),
"$html_el>\n";
}
# else print as text
# add indents at start and at every newline
else {
my ($indent);
$indent = '';
for (1..$indent_level)
{ $indent .= $indent_tab }
$str =~ s|\n|\n$indent|gs;
$str = $indent . $str;
print $fh $str, $newline;
}
ismem($fh) and return $fh->mem();
}
#
# printer
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# web_indent
#
# constant: how many pixels for a single web indent
use constant WEB_INDENT => 20;
sub web_indent {
my ($indent);
if ($indent_level) {
$indent =
'margin-left:' .
($indent_level * WEB_INDENT) .
';';
}
else {
$indent = '';
}
return $indent;
}
#
# web_indent
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showhash
#
=head2 showhash
Displays the keys and values in a hash. Input is either a single hash
reference or a regular hash. The key=values pairs are sorted by the names
of the keys.
So, for example, the following code:
my %hash = (
Larry => 'curly headed guy',
Curly => 'funny bald guy',
Moe => 'guy in charge',
);
showhash %hash;
Produces the following output. Notice that the keys are in alphabetic order:
---------------------------------------
Curly = funny bald guy
Larry = curly headed guy
Moe = guy in charge
---------------------------------------
This code, using a hash reference, produces exactly the same output:
my $hash = {
Larry => 'curly headed guy',
Curly => 'funny bald guy',
Moe => 'guy in charge',
};
showhash $hash;
If the hash is empty, then that fact is output. So, this code:
showhash {};
produces this output:
---------------------------------------
[empty hash]
---------------------------------------
If an undef value is sent instead of a hashref, then that fact is displayed
instead of a hash. For example consider the following code that uses
a variable that is undef:
my ($hash);
showhash $hash;
That code produces this output:
---------------------------------------
Only one element input and it was undefined
---------------------------------------
Optional arguments only come into play if the first argument is a hashref.
B title => "string"
If this option is sent, the string is displayed at the top of the
display of the hash values. So this code:
my $hash = {
Larry => 'curly headed guy',
Curly => 'funny bald guy',
Moe => 'guy in charge',
};
showhash $hash, title=>'Stooges';
produces this output:
--- Stooges ---------------------------------
Curly = funny bald guy
Larry = curly headed guy
Moe = guy in charge
---------------------------------------------
B line_cut => 1
If the C option is sent, then each value is truncated after the first
newline if there is one. The fact that there is more output is mentioned. So
the following code:
my $hash = {
Larry => "curly\nheaded guy",
Curly => "funny\nbald guy",
Moe => "guy\nin charge",
};
showhash $hash, line_cut =>1;
produces this output.
---------------------------------------
Curly = funny [more lines...]
Larry = curly [more lines...]
Moe = guy [more lines...]
---------------------------------------
Several other options do exactly the same thing: linecut, line_chop, and first_line.
=cut
sub showhash {
showstuff() or return '';
# HTML
if (inweb())
{ return showhashhtml(@_) }
# plain text
else
{ return showhashplain(@_) }
}
#
# showhash
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showhashhtml
#
# Private sub. Displays the keys and values in a hash, formatted for HTML.
#
# Code in this sub is not indented because it uses a lot of here documents.
#
sub showhashhtml {
my %myhash;
my $maxkey = 0;
my $maxval = 0;
my $fh = getfh(wantarray);
my $indent = web_indent();
my (%opts, $linecut, @keys, $id_prefix, $table_tag);
# open table
$table_tag = STDTABLE;
$table_tag =~ s|STYLE|$indent|s;
print $fh $table_tag;
# special case: only one element and it's undefined
if ( (@_ == 1) && (! defined($_[0])) ) {
print $fh "| Only element input and it was undefined |
\n";
return;
}
if (ref $_[0]){
%myhash = %{$_[0]};
%opts = @_[1..$#_];
$linecut = $opts{'line_cut'} || $opts{'linecut'} || $opts{'line_chop'} || $opts{'first_line'};
}
else {
%myhash = @_;
}
if (ref $_[0]) {
my $title;
if (defined $opts{'title'})
{$title = $opts{'title'}}
else
{$title = $_[0]}
print $fh '| ', htmlesc($title), " |
\n"
}
print $fh <<"(TABLETOP2)";
| key |
|
value |
(TABLETOP2)
# get array of keys
@keys = keys(%myhash);
# set ID prefix if opted to do so
$id_prefix = $opts{'id_prefix'};
# sort keys unless sort param is sent as false
if (default $opts{'sort'}, 1) {
@keys = sort { lc($a) cmp lc($b) } @keys;
}
# loop through keys, outputting key=value pairs
foreach my $key (@keys) {
my $val = $myhash{$key};
if ($linecut){
$val =~ s|\n.*||s;
$val .= '...';
}
# $id_prefix
print $fh
'',
'| ', htmlesc($key), ' | ',
' | ',
'';
if (defined $id_prefix) {
print $fh
'',
}
print $fh htmlesc($val);
if (defined $id_prefix) {
print $fh '',
}
print $fh ' | ', "
\n";
}
# if empty hash, output that fact
if (! @keys) {
print $fh
'',
'| empty hash | ',
"
\n";
}
# close table
print $fh "\n\n";
ismem($fh) and return $fh->mem;
return '';
}
#
# showhashhtml
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showhashplain
#
# Private sub. Displays the keys and values in a hash, formatted using plain text.
#
sub showhashplain {
my %myhash;
my $maxkey = 0;
my $maxval = 0;
my $fh = getfh(wantarray);
my (%opts, $skipempty, %showfields, $linecut, $divider);
# get title
if (ref $_[0]) {
%myhash = %{$_[0]};
%opts = @_[1..$#_];
if (defined $opts{'title'})
{$divider = '--- ' . $opts{'title'} . ' ---------------------------------'}
}
if (! defined $divider)
{ $divider = '---------------------------------------' }
add_indents($fh);
print $fh $divider, "\n";
$divider =~ s|.|\-|gs;
# special case: only one element and it's undefined
if ( (@_ == 1) && (! defined($_[0])) ) {
add_indents($fh);
print $fh "Only one element input and it was undefined\n";
add_indents($fh);
print $fh $divider, "\n\n";
ismem($fh) and return $fh->mem();
return '';
}
# if first element is a reference, assume it's the hash to be displayed
if (ref $_[0]){
$linecut = $opts{'line_cut'} || $opts{'line_chop'} || $opts{'first_line'};
$skipempty = $opts{'skipempty'};
if (defined $opts{'showfields'}) {
if (ref $opts{'showfields'})
{@showfields{@{$opts{'showfields'}}} = ()}
else
{$showfields{ $opts{'showfields'} } = ()}
}
}
# else all arguments form hash to be displayed
else {
%myhash = @_;
}
# if empty hash
if (! keys(%myhash)) {
add_indents($fh);
print $fh "[empty hash]\n";
}
# not empty hash
else {
my ($keywidth, @keys);
$keywidth = longestkey(\%myhash);
@keys = keys(%myhash);
# sort keys unless sort param is sent as false
if (default $opts{'sort'}, 1) {
@keys = sort { lc($a) cmp lc($b) } @keys;
}
# @keys = sort { lc($a) cmp lc($b) } @keys;
KEYLOOP:
foreach my $key (@keys) {
# if there's a specified list of fields to show
if ( %showfields && (! exists $showfields{$key}) )
{next KEYLOOP}
my $value = $myhash{$key};
# if we should skip empty values
if (
$skipempty &&
(
(! defined $value) ||
($value !~ m|\S|s)
)
)
{ next KEYLOOP }
# linecut
if ($linecut) {
if ( (defined $value) && ($value =~ s|[\r\n].*||s) )
{ $value .= ' [more lines...]' }
}
add_indents($fh);
print $fh
spacepad($key, $keywidth), ' = ',
define_show($value), "\n";
}
}
add_indents($fh);
print $fh $divider, "\n\n";
ismem($fh) and return $fh->mem();
return '';
}
#
# showhashplain
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showarr, showarray
#
=head2 showarr, showarray
Displays the values of an array. c and c
Each element is displayed in a table row (in L)
or on a separate line (in text mode).
If C receives exactly one argument, and if that item is an array
reference, then the routine assumes that you want to display the elements in
the referenced array. Therefore, the following blocks of code display the
same thing:
showarray @myarr;
showarray \@myarr;
=cut
sub showarray {
showstuff() or return '';
my (@arr) = @_;
my $fh = getfh(wantarray);
# if first and only element is an array ref, use that as full array
if ( (@arr == 1) && UNIVERSAL::isa($arr[0], 'ARRAY') )
{@arr = @{$arr[0]}}
#------------------------------------------------------
# HTML
#
if (inweb()) {
print $fh
qq|\n
\n|,
qq|| array |
|;
foreach my $el (@arr){
print $fh '| ';
if (defined $el){
if ($el =~ m|\S|)
{ print $fh htmlesc($el) }
else
{ print $fh 'no content string' }
}
else {
print $fh 'undefined';
}
print $fh " |
\n";
}
print $fh qq|
\n\n|;
}
#
# HTML
#------------------------------------------------------
#------------------------------------------------------
# text
#
else {
my $line = "------------------------------------\n";
my ($firstdone);
add_indents($fh);
print $fh $line;
foreach my $el (@arr){
add_indents($fh);
print $fh define_show($el), "\n";
$firstdone = 1;
}
if (! $firstdone) {
add_indents($fh);
print $fh "[empty array]\n";
}
add_indents($fh);
print $fh $line;
}
#
# text
#------------------------------------------------------
ismem($fh) and return $fh->mem();
}
sub showarr{showarray(@_)}
#
# showarr, showarray
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showarrdiv
#
=head2 showarraydiv
Works just like C, except that in text mode displays a solid line between
each element of the array.
=cut
sub showarraydiv {
showstuff() or return '';
my (@arr) = @_;
my $fh = getfh(wantarray);
if ( (@arr == 1) && UNIVERSAL::isa($arr[0], 'ARRAY') )
{@arr = @{$arr[0]}}
#------------------------------------------------------
# HTML
#
if (inweb()) {
print $fh
"\n
\n",
"| array |
";
foreach my $el (@arr)
{print $fh '| ', htmlesc($el), " |
\n"}
print $fh "
\n\n";
}
#
# HTML
#------------------------------------------------------
#------------------------------------------------------
# text
#
else {
my $line = "------------------------------------\n";
my ($firstdone);
print $fh $line;
foreach my $el (@arr){
#if (defined $el)
# {print $fh $el, "\n"}
#else
# {print $fh "[undef]\n"}
print $fh define_show($el);
print $fh $line;
$firstdone = 1;
}
if (! $firstdone)
{print $fh "[empty array]\n", $line}
}
#
# text
#------------------------------------------------------
ismem($fh) and return $fh->mem();
}
sub showarrdiv{showarraydiv(@_)}
#
# showarraydiv
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showscalar
#
=head2 showscalar
Outputs the value of a scalar. The name is slightly innaccurate: you can
input an array. The array will be joined together to form a single scalar.
Actually, I hardly ever use C, but it seemed unbalanced to have
C and C without C.
=cut
sub showscalar {
showstuff() or return '';
my (@arr) = @_;
my $fh = getfh(wantarray);
#------------------------------------------------------
# HTML
#
if (inweb()) {
print $fh
"\n
\n",
"| scalar |
|---|
";
foreach my $el (@arr)
{print $fh htmlesc($el)}
print $fh " |
\n\n";
}
#
# HTML
#------------------------------------------------------
#------------------------------------------------------
# text
#
else {
print $fh "------------------------------------\n";
if (@arr) {
print $fh
join(
'',
map {define_show($_)}
sort(
{
define_show($a) cmp
define_show($b)
} @arr
)
),
"\n";
}
# {print $fh join('', map {defined($_) ? $_ : '[undef]'} sort({(defined($a) ? $a : '') cmp (defined($b) ? $b : '')} @arr)), "\n"}
else {
print $fh "[no elements]\n";
}
print $fh "------------------------------------\n";
}
#
# text
#------------------------------------------------------
ismem($fh) and return $fh->mem();
}
sub showsca{showscalar(@_)}
#
# showscalar
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showcgi
#
=head2 showcgi
Displays the CGI parameter keys and values. This sub always outputs HTML.
There are several optional parameters, described in the following sections.
B q
The optional parameter C, may be a CGI query object:
my $query = CGI->new();
showcgi q => $query;
If C is not sent, then a CGI object is created on the fly.
B skipempty
If the optional parameter C is true:
showcgi skipempty => 1;
then CGI params that are empty (i.e. do not have
at least one non-space character) are not displayed.
B skip
C sets a list of parameters to not display. For example, if you don't
want to see the C or C params, then call showcgi like this:
showcgi skip => ['choices', 'toppings'];
Single item lists can be passed in directly without putting them in
an anonymous array:
showcgi skip => 'choices';
=cut
sub showcgi {
showstuff() or return '';
my ($q, %opts) = @_;
my (@keys, $fh, $skipempty, %skip, $table_tag);
my $indent = web_indent();
# $q = $opts{'r'} || $opts{'q'} || $opts{'cgi'} || CGI->new();
$skipempty = $opts{'skipempty'};
if (defined $opts{'skip'})
{@skip{asarr($opts{'skip'})} = ()}
@keys = sort $q->param;
$fh = getfh(wantarray);
# open table
$table_tag = STDTABLE;
$table_tag =~ s|STYLE|$indent|s;
print $fh $table_tag;
# title
print $fh <<"(TABLETITLE)";
| CGI params |
(TABLETITLE)
# special case: no elements
if (! @keys) {
print $fh "| No params |
\n";
return;
}
print $fh <<"(TABLETOP2)";
| key |
|
value |
(TABLETOP2)
PARAMLOOP:
foreach my $key (@keys){
my @vals = $q->param($key);
if (exists $skip{$key})
{next PARAMLOOP}
if ($skipempty && @vals <= 1) {
my $val = $vals[0];
if ( (! defined $val) || ($val !~ m|\S|) )
{next PARAMLOOP}
}
print $fh
'| ',
htmlesc($key),
' | | ';
if (@vals > 1) {
# open table
$table_tag = STDTABLE;
$table_tag =~ s|STYLE||s;
print $fh $table_tag;
foreach my $val (@vals) {
print $fh
' |
| ',
htmlesc($val),
" |
\n";
}
print "\n";
}
else {
print $fh
htmlesc($vals[0]);
}
print $fh "\n";
}
print $fh "\n\n";
ismem($fh) and return $fh->mem;
return '';
}
#
# showcgi
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showref
#
=head2 showref($ref, %options)
Displays a hash, array, or scalar references, treeing down through other
references it contains. So, for example, the following code:
my $ob = {
name => 'Raha',
email => 'raha@idocs.com',
friends => [
'Shalom',
'Joe',
'Furocha',
],
};
showref $ob;
produces the following output:
/-----------------------------------------------------------\
friends =
ARRAY
Shalom
Joe
Furocha
email = raha@idocs.com
name = Raha
\-----------------------------------------------------------/
The values of the hash or arrays being referenced are only displayed once, so
you're safe from infinite recursion.
There are several optional parameters, described in the following sections.
B maxhash
The C option allows you to indicate the maximum number of hash
elements to display. If a hash has more then C elements then none of
them are displayed or recursed through, and instead an indicator of how many
elements there are is output. So, for example, the following command only
displays the hash values if there are 10 or fewer elements in the hash:
showref $myob, maxhash=>10;
If C is not sent then there is no maximum.
B maxarr
The C option allows you to indicate the maximum number of array
elements to display. If an array has more then C elements then none
of them are displayed or recursed through, and instead an indicator of how
many elements there are is output. If C is not sent then there is no
maximum.
B depth
The C option allows you to indicate a maximum depth to display in the
tree. If C is not sent then there is no maximum depth.
B skip
A list of hash elements to skip. Only applies to the top element and only if
it is a hash.
B skipall
Works the same as C, but applies to all hashes in the structure, not
just the top-level hash.
=cut
sub showref {
showstuff() or return '';
my ($ref, %opts) = @_;
my ($indentnum, $indent, $type, $tab, %skip, $finalfh);
my $fh = $opts{'fh'} || getfh(wantarray);
# hash of keys to skip
@skip{asarr(delete $opts{'skip'})} = ();
@skip{asarr($opts{'skipall'})} = ();
# set some variables
$indentnum = $opts{'indent'};
$indentnum ||= 0;
$tab = ' ';
$indent = $tab x $indentnum;
$opts{'done'} ||= {};
$type = "$ref";
$type =~ s|^[^\=]*\=||;
$type =~ s|\(.*||;
if (inweb())
{print $fh qq|\n|}
if (! $indent)
{print $fh "/----------------------------------------------------------------------------\\\n"}
if ($type eq 'HASH') {
# if we've recursed to the maximum level
if (
$opts{'indent'} &&
($opts{'indent'}>1) &&
$opts{'maxhash'} && (keys(%{$ref}) > $opts{'maxhash'}) ) {
my $count = keys %{$ref};
print $fh
$indent, '[', $count, ' hash element',
($count>1 ? 's' : ''), "]\n";
}
# else we haven't recursed to the maximum level
else {
if ($opts{'labelself'}) {
print $fh $indent, "HASH\n";
$indentnum++;
$indent .= $tab;
}
ELLOOP:
while ( my($n, $v) = each(%{$ref}) ) {
exists($skip{$n}) and next ELLOOP;
print $fh $indent, $n, ' = ';
if (ref $v) {
if ( $opts{'depth'} ? ($opts{'depth'} >= $indentnum) : 1 ) {
if ($opts{'done'}->{$v})
{print $fh "[redundant]\n"}
else {
$opts{'done'}->{$v} = 1;
print $fh "\n";
showref($v, %opts, done=>$opts{'done'}, indent=>$indentnum+1, fh=>$fh)
}
}
else
{print $fh $v}
}
#elsif (defined $v)
# {print $fh $v, "\n"}
#else
# {print $fh "[undef]\n"}
else {
print $fh define_show($v);
}
}
}
}
elsif ($type eq 'ARRAY') {
print $fh $indent, "ARRAY\n";
if ($opts{'maxarr'} && (@{$ref}) > $opts{'maxarr'} ) {
print
$indent, $tab, '[', scalar(@{$ref}), ' element',
(@{$ref}>1 ? 's' : ''), "]\n";
}
else {
my ($firstdone);
foreach my $v ( @{$ref} ) {
if (ref $v) {
if ( $opts{'depth'} ? ($opts{'depth'} >= $indentnum) : 1 ) {
if ($opts{'done'}->{$v})
{print $fh $indent, $tab, '[redundant]'}
else {
$opts{'done'}->{$v} = 1;
if ($firstdone)
{print $fh "\n"}
else
{$firstdone = 1}
showref($v, %opts, done=>$opts{'done'}, indent=>$indentnum+1, labelself=>1, fh=>$fh)
}
}
else
{print $fh $indent, $tab, $v}
}
#elsif (defined $v)
# {print $fh $indent, $tab, $v, "\n"}
#else
# {print $fh $indent, $tab, "[undef]\n"}
else {
print $fh $indent, $tab, define_show($v), "\n"
}
}
}
}
if (! $indent)
{print $fh "\\----------------------------------------------------------------------------/\n\n"}
if (inweb())
{print $fh "\n"}
ismem($fh) and return $fh->mem();
}
#
# showref
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# getfh
#
# Private sub. Returns a file handle that is either STDOUT or STDERR (depending
# on the value of the global variable $Debug::ShowStuff::out), or a MemHandle
# object that will be used to return a string to the caller of the function
# that called getfh.
#
sub getfh {
my ($wa) = @_;
my ($fh);
# if explicit context
if ($always_void)
{ undef $wa }
# if called in void context, outputs to STDOUT,
# otherwise returns string
if (defined $wa) {
require MemHandle;
$fh = MemHandle->new('');
}
else {
$fh = $out;
}
# return file handle
return $fh;
}
#
# getfh
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# ismem
#
# This private function returns true if the given file handle
# is a MemHandle filehandle.
#
sub ismem {
return UNIVERSAL::isa($_[0], 'MemHandle');
}
#
# ismem
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# spacepad
#
# This private function returns the given string padded with
# the indicated number of spaces. The spaces are added to the right
#
sub spacepad {
my ($str, $width) = @_;
return sprintf("%-${width}s", $str);
}
#
# spacepad
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# longestkey
#
# This private function returns the length of the longest key
# in the given hash. Only pass in hash references, not hashes.
#
sub longestkey {
my ($hash) = @_;
my $max = 0;
foreach my $key (keys %$hash) {
if (length($key) > $max)
{ $max = length($key) }
}
return $max;
}
#
# longestkey
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# asarr
#
# Private function. Allows an optional argument to be passed as
# either an array reference or as a single item. Always returns
# an array reference in scalar context and an array in array context.
#
sub asarr {
my $arg = shift;
my @rv;
if (ref $arg)
{@rv = @$arg}
elsif (defined $arg)
{@rv = $arg}
wantarray and return @rv;
return \@rv;
}
#
# asarr
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# printhr
#
=head2 printhr
Prints a horizontal rule. Handy for dividing up multiple println's.
In text mode, the horizontal rule is a set of 80 dashes. In
L, the output is either a
element or a
element, depending on the title option (see "title" below).
So, for example, the following line outputs a simple horizontal rule:
B title
If the Coption is sent, the title is embedded in the horizontal rule.
So, for example, the following code produces a horizontal rule with with the
string "test" embedded in it:
printhr title=>'test';
If only one param is sent, it is assumed that param is the title. So the'
following code produces exactly the same thing as the example above:
printhr 'test';
In web mode, a title changes the HTML element that is output. If no title is
given then printhr outputs an
element. If a title is given the output is
C element with the title as the content. The
element has a gray
background and a black border.
B dash
If the Coption is sent, the given character is used as the separator.
This param only applies to text mode;
=cut
sub printhr {
showstuff() or return '';
my (%opts, $title);
my $fh = getfh(wantarray);
# single argument: title
if (@_ == 1) {
$title = $_[0];
}
else {
%opts = @_;
$title = $opts{'title'} || $opts{'msg'};
}
# web mode
if (inweb()) {
my $indent = web_indent();
if (defined $title) {
print $fh <<"(HTML)";
@{ [ htmlesc($title) ] }
(HTML)
}
else {
print $fh qq|
\n|;
}
}
# else text mode
else {
# set dash
my $dash = default($opts{'dash'}, '-');
# add vertical space
if ($opts{'vspace'})
{ print "\n" }
add_indents($fh);
if (defined $title) {
print $fh
($dash x 3),
" $title ",
($dash x (75 - length($title))),
"\n";
}
else {
print $fh ($dash x 80), "\n";
}
}
}
#
# printhr
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# text_window_size
#
sub text_window_size {
my (@dimensions, %rv);
# load necessary module
require Term::ReadKey;
# get sizes
# ($wchar, $hchar, $wpixels, $hpixels) = GetTerminalSize();
@dimensions = Term::ReadKey::GetTerminalSize();
# set hash
$rv{'chars_wide'} = $dimensions[0];
$rv{'chars_tall'} = $dimensions[1];
$rv{'pixels_wide'} = $dimensions[2];
$rv{'pixels_tall'} = $dimensions[3];
# return
return \%rv;
}
#
# text_window_size
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# add_indents
#
# Private function
#
sub add_indents {
my ($fh) = @_;
for (1..$indent_level) {
print $fh $indent_tab;
}
}
#
# add_indents
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# printval
#
# I don't even remember what this function is for, so for now I'm commenting
# it out.
#
# sub printval {
# showstuff() or return '';
#
# my ($val) = @_;
# my $fh = getfh(wantarray);
#
# # get value to print
# $val = define_show($val);
#
# if (inweb())
# { $val = htmlesc($val) }
#
# print $fh $val;
#
# ismem($fh) and return $fh->mem();
# }
#
# printval
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# preln
#
=head2 preln
Outputs the given values inside a element. If not in a web environment,
works just like preln.
=cut
sub preln {
showstuff() or return '';
if (! inweb())
{ return println(@_) }
my ($str);
my $fh = getfh(wantarray);
# special case: no inputs
if (! @_ )
{$str = 'empty'}
# special case: just one element and it's undefined
elsif ( (@_ == 1) && (! defined $_[0]) )
{$str = 'undefined'}
# else lump them all together
else
{$str = htmlesc(join('', @_))}
print $fh
'',
$str,
"
\n";
ismem($fh) and return $fh->mem();
}
#
# preln
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# dieln
#
=head2 dieln
Works like the C command, except it always adds an end-of-line to the
input array so you never get those "at line blah-blah-blah" additions.
=cut
sub dieln {
my ($str);
$str = join('', map {define_show($_)} @_);
die $str . "\n";
}
#
# dieln
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# devexit
#
=head2 devexit
Works like C except it prepends 'dev exit: ' to the end of the string.
If no string is sent, just outputs "dev exit".
=cut
sub devexit {
my (@str) = @_;
my ($rv);
if (defined $str[0])
{ $rv = 'dev exit: ' . join('', @str) }
else
{ $rv = 'dev exit' }
# exit in web, die in commandline
if ( inweb(strict=>1) ) {
println $rv;
exit;
}
else {
dieln $rv;
}
}
#
# devexit
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# diearr
#
=head2 diearr
Displays an array, then dies using C.
=cut
sub diearr {
my $err = shift;
showarr @_;
dieln $err;
}
#
# diearr
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# pressenter
#
=head2 pressenter
For use at the command line. Outputs a prompt to "press enter to continue",
then waits for you to do exactly that.
=cut
sub pressenter {
my ($msg) = @_;
my ($fh);
$msg ||= 'press enter to continue';
$msg =~ s|\s*$||s;
# load IO::Handle
require IO::Handle;
# output message
println $msg;
# flush output file handle
$fh = getfh(wantarray);
$fh->flush();
# wait
;
}
#
# pressenter
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# confirm
#
=head2 confirm
Prompts the user for a y or n. Exits quietly if y is pressed.
=cut
sub confirm {
my ($msg) = @_;
$msg ||= 'continue?';
$msg =~ s|\s*$| |s;
RESPONSELOOP:
while (1) {
my ($response);
print $msg;
$response = ;
$response =~ s|^\s*(.).*|$1|s;
$response = lc($response);
if ($response eq 'y')
{ return 1 }
elsif ($response eq 'n')
{ exit }
}
}
#
# confirm
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# httpheader
#
=head2 httpheader
Outputs a text/html HTTP header. Not useful if you're using mod_perl.
=cut
sub httpheader {
my (%opts) = @_;
if (wantarray)
{return "Content-type:text/html\n\n"}
my $fh = $opts{'fh'} || getfh(wantarray);
print $fh "Content-type:text/html\n\n";
}
sub httpheaders{return httpheader(@_)}
#
# header
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showstderr
#
=head2 showstderr
This function allows you to see, in the web page produced by a CGI, everything
the CGI output to STDERR.
To use C, assign the return value of the function to a variable
that is scoped to the entire CGI script:
my $stderr = showstderr();
You need not do anything more with that variable. The object reference by
your variable holds on to everything output to both STDOUT and STDERR. When
the variable goes out of scope, the object outputs the STDOUT content with the
STDERR content at the top of the web page.
=cut
sub showstderr {
return Debug::ShowStuff::ShowStdErr->new(@_);
}
#
# showstderr
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# forceenv
#
=head2 forcetext, forceweb, forcenone
By default, Debug::Showstuff guesses if it should be in
L. These functions are for when you
want to explicitly tell Debug::ShowStuff what mode it should be in.
C forces text mode. C forces web mode. C
tells Debug::Showstuff that you don't want to force either mode.
=cut
sub forcetext {
forceenv(0);
}
sub forceweb {
forceenv(1);
}
sub forcenone {
forceenv(undef);
}
sub forceenv {
$forceweb = $_[0];
}
#
# forceenv
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# inweb
#
=head2 inweb
Returns a guess on if we're in a L. The
guess is pretty simple: if the environment variable C is true (in
the Perlish sense) then this function returns true.
If the global C<$Debug::ShowStuff::forceenv> is defined, this function returns
the value of C<$Debug::ShowStuff::forceenv>.
=cut
sub inweb {
my (%opts) = @_;
unless ($opts{'strict'}) {
if (defined $forceweb)
{ return $forceweb }
}
return ($ENV{'REQUEST_URI'} || $ENV{'SERVER_NAME'});
}
#
# inweb
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# output_to_file
#
=head2 output_to_file($path)
Sends Debug::ShowStuff output to a file instead of to STDOUT or STDERR. The
value of this function must be assigned to a variable or it has no effect.
Don't do anything with the returned value... it is NOT a file handle. The
returned value is an object that, when it goes out of scope, closes the output
file handle.
For example, the following code will output to three files names
Larry.txt, Curly.txt, and Moe.txt:
foreach my $name (qw[Larry Curyl Moe]) {
my $output = output_to_file("$name.txt");
println $name;
println 'length: ', length($name);
}
=cut
sub output_to_file {
my ($path, %opts) = @_;
my ($fh, $arrows, $closer);
if (! defined wantarray) {
my $msg =
"Do not call outputfile in void context. Save the return value to a scalar.\n" .
"REMINDER: The return value is NOT a filehandle, it is just an object that, when\n" .
"it goes out of scope, closes the global filehandle.\n";
croak $msg;
}
# if path is a reference, assume it's a filehandle object
if (ref $path) {
$fh = $path;
}
# else open filehandle to path
else {
require FileHandle;
# open new or open append
if ($opts{'append'})
{ $arrows = '>>' }
else
{ $arrows = '>' }
$fh = FileHandle->new("$arrows$path");
$fh->autoflush(1);
}
setoutput($fh);
$closer = Debug::ShowStuff::CloseGlobalHandle->new($fh);
return $closer;
}
#
# output_to_file
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# setoutput
#
=head2 setoutput
Sets the default output handle. By default, routines in C
output to STDOUT. With this command you can set the default output to STDERR,
back to STDOUT, to a filehandle you specify, or to a
Debug::ShowStuff::SeparatePrint file handle.
The following command sets default output to STDERR:
setoutput 'stderr';
This command sets output back to STDOUT:
setoutput 'stdout';
This command sends output to a file handle of your choice:
setoutput $fh;
This command sends output to a Debug::ShowStuff::SeparatePrint file handle.
This option is a good way to create a simple log file. When you print to this
type of file handle, the file is opened separately just for that write, an
exclusive lock on the file is obtained, and the end of the file is sought.
Note that the next parameter must be the path to the output file:
setoutput 'separateprint', $file_path;
B new
If the C parameter is true, then the output file is open in non-append
mode, which means any existing contents are removed.
=cut
sub setoutput {
my $outname = shift;
# if reference, assume it's filehandle
# if ( UNIVERSAL::isa($outname, 'FileHandle') ) {
if ( ref $outname ) {
$out = $outname;
$forceweb = 0;
}
# STDOUT
elsif (lc($outname) eq 'stdout') {
$out = *STDOUT;
}
# STDERR
elsif (lc($outname) eq 'stderr') {
$out = *STDERR;
$forceweb = 0;
}
# Debug::ShowStuff::SeparatePrint
elsif (lc($outname) eq 'separateprint') {
$out = Debug::ShowStuff::SeparatePrint->new(@_);
}
# else don't know
else {
croak "do not know this type of output: $outname";
}
}
#
# setoutput
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# fixundef
#
=head2 fixundef
Takes a single argument. If that argument is undefined, returns an
empty string. Otherwise returns the argument exactly as it is.
=cut
sub fixundef {
defined($_[0]) or return '';
return $_[0];
}
#
# fixundef
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# findininc
#
=head2 findininc
Given one or more file names, searches @INC for where they are located.
Returns an array of full file names.
=cut
sub findininc {
my (@filenames) = @_;
my (@rv);
foreach my $dir (@INC) {
foreach my $filename (@filenames) {
my $path = "$dir/$filename";
if (-e $path)
{push @rv, $path}
}
}
if (! defined(wantarray))
{showarr @rv}
else
{return @rv}
}
#
# findininc
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# define_show
#
# Private function. If the given value is undef, returns the string "[undef]".
# If it is an empty string, returns "[empty string]". Otherwise the original
# string is returned.
#
use overload;
sub define_show {
my ($val) = @_;
# if overloaded object, get return value and
# concatenate with string (which defines it).
if (ref $val) {
if (my $func = overload::Method($val, '""'))
{ $val = &$func($val) }
}
if (defined $val) {
if ($val eq '')
{ $val = '[empty string]' }
#elsif ($val =~ m|^\s+$|s)
# { $val = '[space-only string]' }
}
else {
$val = '[undef]';
}
return $val;
}
#
# define_show
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showtainted
#
=head2 showtainted(@values)
Given an array of values, shows which are tainted and which are not. If the
first argument is a hashref, displays the tainted status of each element value.
=cut
sub showtainted {
my (@vals) = @_;
# require Taint;
require Scalar::Util;
# if first value is a hashref, show as hash
if (UNIVERSAL::isa $vals[0], 'HASH') {
my $hashref = $vals[0];
my $show = {};
while ( my($key, $val) = each(%$hashref) ) {
my ($tainted);
if (Scalar::Util::tainted($val))
{ $tainted = 'tainted' }
else
{ $tainted = 'not tainted' }
$show->{$key} = $tainted;
}
showhash $show;
}
# loop through values
else {
foreach my $val (@vals) {
my ($tainted);
if (Scalar::Util::tainted($val))
{ $tainted = 'tainted: ' }
else
{ $tainted = 'not tainted: ' }
$val = "$tainted$val";
}
# show values
showarr @vals;
}
}
#
# showtainted
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showsth
#
=head2 showsth
Outputs a table of all rows in the given DBI statement handle. Note that
this function "uses up" the statement handle.
=cut
sub showsth {
showstuff() or return '';
if (inweb())
{ return showsth_web(@_) }
else
{ return showsth_text(@_) }
}
#
# showsth
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showsql
#
=head2 showsql
showsql output the results of an SQL statement. showsql takes three
parameters: the database handle, the SQL statement, and an array-ref of
parameters for the SQL statement.
=cut
sub showsql {
my $adbh = shift;
my $sql = shift;
my $params = shift;
return showsth($adbh, sql=>$sql, params=>$params, @_);
}
#
# showsql
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# explainsql
#
=head2 explainsql
explainsql outputs the result of an SQL EXPLAIN call. This function works much
like showsql. The parameters are the database handle, the SQL statement, and
an array-ref of SQL parameters. explainsql prepends "EXPLAIN ANALYZE" to your
SQL, runs the statement, then outputs the results.
I have only used explainsql with PostGresql. I would be interested hear about
how it works with other database management systems and how it might be
improved to work in those environments.
=cut
sub explainsql {
my ($adbh, $sql, $params_org) = @_;
my (@params, $row);
# if any params, put them in @params array
if (defined $params_org) {
if (UNIVERSAL::isa $params_org, 'ARRAY')
{ @params = @{$params_org} }
else
{ @params = $params_org }
}
# modify $sql
$sql = qq|EXPLAIN ANALYZE\n$sql|;
# get results row
$row = $adbh->selectrow_hashref($sql, undef, @params);
# if we got a row
if ($row) {
preln $sql;
preln
qq|QUERY PLAN\n----------------------------------------\n|,
$row->{'QUERY PLAN'};
}
# else error
else {
println 'error getting SQL explain: ', $DBI::errstr;
}
}
#
# explainsql
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# get_sth
# Private method
#
sub get_sth {
my ($sth, %opts) = @_;
# if $sth is a db, create a statement handle
if ( UNIVERSAL::isa($sth, 'DBI::db') ) {
my ($dbh, $sql, @params);
$sql = $opts{'sql'};
$dbh = $sth;
# must have sql param
if (! $sql)
{ croak 'If $sth is a database handle, must have sql param' }
# if any params, put them in @params array
if ($opts{'params'}) {
if (UNIVERSAL::isa $opts{'params'}, 'ARRAY')
{ @params = @{$opts{'params'}} }
else
{ @params = $opts{'params'} }
}
# create statement handle
$sth = $dbh->prepare($sql);
$DBI::err and croak $DBI::errstr;
# execute statement handle
$sth->execute(@params);
$DBI::err and croak $DBI::errstr;
}
# return
return $sth;
}
#
# get_sth
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showsth_text
#
sub showsth_text {
my ($sth, %opts) = @_;
my ($table, @fields, @rows, $out);
require Text::TabularDisplay;
# get statement handle, in case a database handle was actually sent
# KLUDGE: I got the idea that allowing a dbh and an sql statement
# instead of a statement handle would be a good feature, in case I
# didn't actually want to go to the trouble of creating a statement
# handle just to display the results of an SQL statement. The general
# idea is good, but should have been implemented as a separate method.
# There is now a method called showsql, but it still relies on this
# method to tell if a database or statement handle was sent. The kludge
# here is that for now I'm choosing not to clean up the mess.
$sth = get_sth($sth, %opts);
# ensure statement handle is executed
if (! $sth->{'Active'}) {
$sth->execute();
$DBI::err and die $DBI::errstr;
}
# get array of fields
@fields = @{$sth->{'NAME'}};
# create text-table object
$table = Text::TabularDisplay->new(@fields);
# loop through rows
while (my $row_hash = $sth->fetchrow_hashref) {
my (@row_arr);
foreach my $field (@fields)
{ push @row_arr, define($row_hash->{$field}) }
push @rows, \@row_arr;
}
# populate text table
$table->populate(\@rows);
# output text table
$out = $table->render();
# build header
if (hascontent $opts{'title'}) {
my ($border, $title);
# build top border
$border = $out;
$border =~ s|\n.*||s;
$border =~ s|\+|-|gs;
$border =~ s|^\-|+|gs;
$border =~ s|\-$|+|gs;
# build title
$title = '| ' . crunch($opts{'title'});
$title .= repeat(' ', length($border) - length($title) - 1);
$title .= '|';
# add to output
$out = $border . "\n" . $title . "\n" . $out;
}
# output
println $out;
# success
return '';
}
#
# showsth_text
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showsth_web
#
sub showsth_web {
my ($sth, %opts) = @_;
my ($first_row_done, %field_names, $formats);
tie %field_names, 'Tie::IxHash';
$formats = $opts{'formats'} || {};
# get statement handle, in case a database handle was actually sent
# KLUDGE: I got the idea that allowing a dbh and an sql statement
# instead of a statement handle would be a good feature, in case I
# didn't actually want to go to the trouble of creating a statement
# handle just to display the results of an SQL statement. The general
# idea is good, but should have been implemented as a separate method.
# There is now a method called showsql, but it still relies on this
# method to tell if a database or statement handle was sent. The kludge
# here is that for now I'm choosing not to clean up the mess.
$sth = get_sth($sth, %opts);
# get headers that are to be output first
if ($opts{'headers'}) {
foreach my $field_name (@{$opts{'headers'}})
{ undef $field_names{$field_name} }
}
# open table
print <<'(HTML)';
(HTML)
# output title if available
if (defined $opts{'title'}) {
print
'',
'| ',
$opts{'title'},
" |
\n";
}
# loop through records
while (my $row = $sth->fetchrow_hashref) {
if (! $first_row_done) {
# build list of headers
foreach my $field_name (@{$sth->{'NAME'}}) {
if (! exists $field_names{$field_name})
{ undef $field_names{$field_name} }
}
# open table and header row
print <<'(HTML)';
(HTML)
# output headers
foreach my $field_name (keys %field_names) {
print '| ', htmlesc($field_name), " | \n";
}
# close header row
print <<'(HTML)';
(HTML)
# note table headers have been output
$first_row_done = 1;
}
# open row
print '';
# loop through fields
foreach my $field_name (keys %field_names) {
my ($value, $format);
$value = $row->{$field_name};
$format = $formats->{$field_name} || {};
# |
if ($format->{'align'})
{ print qq| | | }
else
{ print ' | ' }
if (defined $value) {
if ($value eq '') {
print '[empty string]';
}
else {
my $output = htmlesc($value);
# pre
if ($format->{'pre'}) {
$output =~ s|\t| |gs;
$output = "$output ";
}
print $output;
}
}
else {
print '[undef]';
}
print " | \n";
}
# close row
print "
\n";
}
# close table or mention that there were no records
if ($first_row_done) {
print qq|\n|;
}
else {
print qq|| no records |
\n\n|;
}
print qq|
\n|;
}
#
# showsth_web
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# tempshowstuff
#
=head2 tempshowstuff
Temporarily turn showstuff on or off. Create a variable in the lexical scope
where you want the tempoary change, like this:
my $temp = tempshowstuff(1)
When the variable goes out of scope, showstuff will revert back to its previous
state.
=cut
sub tempshowstuff {
return Debug::ShowStuff::TempShowStuff->new();
}
#
# tempshowstuff
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# showisa
#
=head2 showisa
Outputs the ISA hierarchy of an object or class. For example, the following
code outputs the ISA hierarchy for a Xapat object (Xapat is a web server
written in Perl and which uses Net::Server).
$xapat = Xapat->new(%opts);
showisa $xapat;
which outputs:
------------------------------------
Xapat
Net::Server::HTTP
Net::Server::MultiType
Net::Server
------------------------------------
Note that showisa loads Class::ISA, which is available on CPAN.
=cut
sub showisa {
my ($class) = @_;
my (@isas);
# load necessary module
require Class::ISA;
# if $class is an object, get its class
if (ref $class)
{ $class = ref $class }
# get class hierarchy
@isas = Class::ISA::self_and_super_path($class);
# show class hierarchy
showarr @isas;
}
#
# showisa
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# timer
#
=head2 timer
This function is for when you want to display how long it took for your code to
run. Assign the return value of this function to a variable. When the
variable goes out of scope then the difference between the start and end time
is displayed in seconds. For example, the following code:
do {
my $timer = timer();
sleep 3;
};
outputs this:
start timer
duration: 3 seconds
B title
If you send the C option, then that title will be displayed with the
beginning and ending output. For example, this code:
do {
my $timer = timer(title=>'my block');
sleep 3;
};
produces this output:
start timer - my block
duration - my block: 3 seconds
B $timer->elapsed
Returns the difference between when the timer was started and the current
time.
B $timer->silence
Turns off the timer so that it doesn't display anything when it dies.
=cut
sub timer {
my (%opts) = @_;
return Debug::ShowStuff::Timer->new(%opts);
}
#
# timer
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# backtrace
#
sub backtrace {
my (%opts);
# if just one param, assume that's the title
if (@_ == 1)
{ %opts = (title=>@_) }
else
{ %opts = @_ }
if (inweb())
{ return backtrace_web(%opts) }
else
{ return backtrace_text(%opts) }
}
#
# backtrace
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# backtrace_text
# Private subroutine
#
sub backtrace_text {
my (%opts) = @_;
my ($indent, $table, @frames, $render);
# load Devel::Backtrace module
require Text::TabularDisplay;
# indent to make a nice clean backtrace
$indent = indent();
# get frames
@frames = backtrace_frames();
# create text-table object
$table = Text::TabularDisplay->new(qw(line package sub));
# populate text table
$table->populate(\@frames);
# get rendered table
$render = $table->render();
# if there is any message
if (hascontent $opts{'title'}) {
my ($first_line, $line_length, $title);
# build first line of table
$first_line = $render;
$first_line =~ s|\n.*||s;
$first_line =~ s|[^-]|-|gs;
$first_line =~ s|^.|+|gs;
$first_line =~ s|.$|+|gs;
$line_length = length($first_line) - 1;
# build title line
$title = '| ' . $opts{'title'};
while (length($title) < $line_length) {$title .= ' '}
$title .= '|';
# build $render
$render = $first_line . "\n" . $title. "\n" . $render;
}
# output
println $render;
}
#
# backtrace_text
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# backtrace_web
# Private subroutine
#
sub backtrace_web {
my (%opts) = @_;
my ($last_sub, @frames, $render, $fh, $html, $title);
showstuff() or return '';
# get output handle
$fh = getfh(wantarray());
# get frames
@frames = backtrace_frames();
# title row
if (defined $opts{'title'}) {
$title = htmlesc($opts{'title'});
print STDERR 'BACKTRACE: ', $title, "\n";
}
else {
$title = 'stack trace';
}
# start table
$html = <<"(HTML)";
| $title |
| line |
module |
sub |
(HTML)
# loop through frames
foreach my $frame (@frames) {
my $mod = htmlesc($frame->[1]);
# break up module name
$mod =~ s|::|:: |gs;
$html .=
qq|\n| | .
htmlesc($frame->[0]) .
" | \n" .
$mod .
" | \n" .
htmlesc($frame->[2]) .
" | \n
\n\n";
}
# close table
$html .= "\n
";
# note dying if doing so
if ($opts{'die'})
{ println 'die' }
# output
# print $fh '', $html, '';
print $fh $html;
}
#
# backtrace_web
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# backtrace_frames
#
sub backtrace_frames {
my ($trace, @frames, $last_sub);
# create backtrace object
require Devel::StackTrace;
$trace = Devel::StackTrace->new;
# from bottom (least recent) of stack to top.
while (my $frame = $trace->prev_frame) {
my ($sub);
# if the subroutine from the previous iteration exists,
# output it, otherwise empty string
if (defined $last_sub)
{ $sub = $last_sub }
else
{ $sub = '-' }
# now hold on to last sub
$last_sub = $frame->subroutine;
$last_sub =~ s|^.*\:||s;
# add frame data
push @frames, [
$frame->line,
$frame->package,
$sub];
}
# remove last two frames, which are just this subroutine and the caller
pop @frames;
pop @frames;
# pop @frames;
# special case: if called from dietrace, don't include that in output
if ($frames[-1]->[2] eq 'dietrace')
{ pop @frames }
# return
return @frames;
}
#
# backtrace_frames
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# dietrace
#
sub dietrace {
# showstuff(1);
backtrace(@_);
# die
die '[die]';
}
#
# dietrace
#------------------------------------------------------------------------------
###############################################################################
# ShowStdErr
#
package Debug::ShowStuff::ShowStdErr;
use strict;
#---------------------------------------------------------------------------
# new
#
sub new {
my ($class, %opts) = @_;
my $self = bless({}, $class);
# default to HTML environment if environment appears to indicate so
if ( (! defined $opts{'html'}) && $ENV{'REQUEST_URI'})
{$opts{'html'} = 1}
# default certain properties based on html
if ($opts{'html'}) {
defined($opts{'flushtostderr'}) or $opts{'flushtostderr'} = 1;
defined($opts{'stderrfirst'}) or $opts{'stderrfirst'} = 1;
}
# default to output STDERR first
$opts{'stderrfirst'} = exists($opts{'stderrfirst'}) ? $opts{'stderrfirst'} : 1;
# other properties
$self->{'html'} = $opts{'html'};
$self->{'flushtostderr'} = $opts{'flushtostderr'};
$self->{'stderrfirst'} = $opts{'stderrfirst'};
# capture STDERR
$self->{'stderrdata'} = {'str' => []};
open(SAVEERR, ">&STDERR") or warn "Cannot save STDERR: $!\n";
print SAVEERR '';
$self->{'saveerr'} = *SAVEERR;
$self->{'caperr'} = tie(*STDERR, $class . '::HandleOb', $self->{'stderrdata'}, %opts);
# capture STDOUT
if ($opts{'stderrfirst'}) {
$self->{'stdoutdata'} = {'str' => []};
open(SAVESTD, ">&STDOUT") or warn "Cannot save STDOUT: $!\n";
print SAVESTD '';
$self->{'savestd'} = *SAVESTD;
$self->{'capstd'} = tie(*STDOUT, $class . '::HandleOb', $self->{'stdoutdata'}, %opts);
}
# warn() doesn't seem to print to STDERR through Perl.
# Catch that manually.
$self->{'oldwarn'} = $SIG{__WARN__};
$SIG{__WARN__}= sub {print STDERR "@_"};
return $self;
}
#
# new
#---------------------------------------------------------------------------
#---------------------------------------------------------------------------
# displaystderr
#
sub displaystderr {
my ($self) = @_;
# early exit: if there isn't anything in STDERR, do nothing
unless (@{$self->{'stderrdata'}->{'str'}})
{return}
# output to the real STDERR if necessary
if ($self->{'flushtostderr'})
{print STDERR join("\n",@{$self->{'stderrdata'}->{'str'}})}
# output as HTML if set to do so
if ($self->{'html'}) {
print
'',
"
STDERR
\n
\n";
foreach my $line (@{$self->{'stderrdata'}->{'str'}}) {
# escape the HTML
$line =~ s|&|&|g;
$line =~ s|"|"|g;
$line =~ s|'|'|g;
$line =~ s|<|<|g;
$line =~ s|>|>|g;
# output the line
print $line;
}
print "\n
\n\n";
}
# else output as text
else {
print
"========================================================================\n",
"STDERR\n\n",
@{$self->{'stderrdata'}->{'str'}}, "\n",
"========================================================================\n";
}
}
#
# displaystderr
#---------------------------------------------------------------------------
#------------------------------------------------------------------------------
# DESTROY
#
DESTROY {
my ($self) = @_;
#------------------------------------------------
# release handles
#
$SIG{__WARN__} = $self->{'oldwarn'};
undef $self->{'caperr'}; # as documented in "perltie"
untie(*STDERR) or warn("Cannot untie STDERR: $!\n");
# open(STDERR, ">&SAVEERR") or warn("Cannot restore STDERR: $!\n");
*SAVEERR = $self->{'saveerr'};
open(STDERR, ">&SAVEERR") or warn("Cannot restore STDERR: $!\n");
if ($@)
{die "$@"}
# if also capturing stdout
if ($self->{'stdoutdata'}) {
undef $self->{'capstd'}; # as documented in "perltie"
untie(*STDOUT) or warn("Cannot untie STDOUT: $!\n");
*SAVESTD = $self->{'savestd'};
open(STDOUT, ">&SAVESTD") or warn("Cannot restore STDOUT: $!\n");
if ($@)
{die "$@"}
}
#
# release handles
#------------------------------------------------
#------------------------------------------------
# display data
#
if ($self->{'stderrfirst'}) {
# put STDOUT into string
my $stdout = join('', @{$self->{'stdoutdata'}->{'str'}});
# if HTML
if ($self->{'html'}) {
my ($borders);
# pull out headers
# this part is a little kludgy
# I couldn't get the split on the following line
# to work quite right
# ($borders, $stdout) = split(m/(\r\n\r\n)/s, $stdout, 2);
$borders = $stdout;
$borders =~ s/((?:\r\n\r\n)|(?:\n\n)|(?:\r\r)).*//s;
$stdout = substr($stdout, length($borders) + length($1));
# output headers
print $borders, "\n\n";
}
# output stderr
$self->displaystderr;
# if stdout has any content, and it doesn't end in
# a newline, add a newline
if ( length($stdout) && ($stdout !~ m|[\r\n]$|) )
{$stdout .= "\n"}
# output stdout
print $stdout;
}
# else just output STDERR
else {
$self->displaystderr;
}
#
# display data
#------------------------------------------------
}
#
# DESTROY
#------------------------------------------------------------------------------
#
# ShowStdErr
###############################################################################
###############################################################################
# HandleOb
#
package Debug::ShowStuff::ShowStdErr::HandleOb;
use strict;
use Carp;
sub TIEHANDLE {
my($class, $data, %opts) = @_;
my $self= bless( {} , $class);
$self->{'croakonerr'} = $opts{'croakonerr'};
$self->{'data'} = $data;
return($self);
}
sub WRITE {
my($self, $buf, $len, $offset) = @_;
push @{$self->{'data'}->{'str'}}, $buf;
return 1;
}
sub PRINT {
my $self = shift;
# croak if necessary
if ($self->{'croakonerr'})
{croak @_}
push @{$self->{'data'}->{'str'}}, @_;
return 1;
}
sub PRINTF {
my $self = shift;
my $fmt = shift;
# $self->{'data'}->{'str'} .= sprintf($fmt, @_);
push @{$self->{'data'}->{'str'}}, sprintf($fmt, @_);
return 1;
}
sub AUTOLOAD {
}
sub readwarning {
carp "Cannot read from specified filehandle.";
}
#
# HandleOb
###############################################################################
###############################################################################
# CloseGlobalHandle
#
# Private function.
# Objects in this class automatically undefs Debug::ShowStuff::out if it is
# a FileHandle object;
#
# Objects in this class are used by the function output_to_file().
#
package Debug::ShowStuff::CloseGlobalHandle;
use strict;
# use Carp;
#------------------------------------------------------------------------------
# new
#
sub new {
my ($class, $fh) = @_;
my $self = bless {}, $class;
$self->{'fh'} = $fh;
return $self;
}
#
# new
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# DESTROY
#
DESTROY {
my ($self) = @_;
my $global = $Debug::ShowStuff::out;
# if the global output filehandle still exists
if ($global) {
# if it's a filehandle object
if ( UNIVERSAL::isa($global, 'FileHandle') ) {
# if it's the same filehandle object as this object references
if ($global == $self->{'fh'}) {
# print "closing $global\n";
undef $global;
undef $Debug::ShowStuff::out;
Debug::ShowStuff::setoutput('stdout');
}
}
}
}
#
# DESTROY
#------------------------------------------------------------------------------
#
# CloseGlobalHandle
###############################################################################
###############################################################################
# Indent
#
# An object in this class for the purpose of decrementing the current
# indent level when it is destroyed. On destruction is decrements
# $Debug::ShowStuff::indent_level.
#
package Debug::ShowStuff::Indent;
use strict;
#---------------------------------------------------------------------------
# new
#
sub new {
my ($class, %opts) = @_;
my ($increment);
if (defined $opts{'title'}) {
$Debug::ShowStuff::indent_level++;
Debug::ShowStuff::println($opts{'title'});
$increment = 2;
}
else {
$increment = 1;
}
# increment indent level
$Debug::ShowStuff::indent_level++;
my $indent = bless({increment=>$increment, %opts}, $class);
return $indent;
}
#
# new
#---------------------------------------------------------------------------
#------------------------------------------------------------------------------
# DESTROY
#
DESTROY {
my ($self) = @_;
# add bottom space if necessary
if ($self->{'bottom_space'})
{ Debug::ShowStuff::println() }
if ($Debug::ShowStuff::indent_level > 0 ) {
# $Debug::ShowStuff::indent_level--;
$Debug::ShowStuff::indent_level -= $self->{'increment'};
}
}
#
# DESTROY
#------------------------------------------------------------------------------
#
# CloseIndent
###############################################################################
###############################################################################
# Timer
#
# An object in this class for the purpose of showing the amount of time
# something took to run. When the object is initialized it notes the current
# time. When it is destroyed it displays the number of seconds between when
# it was created and when it was destroyed.
#
package Debug::ShowStuff::Timer;
use strict;
#---------------------------------------------------------------------------
# new
#
sub new {
my ($class, %opts) = @_;
my ($timer, $init_title);
# default options
%opts = (init_title=>1, %opts);
# create timer object
$timer = bless({%opts}, $class);
# hold on to start time
$timer->{'start'} = time();
# silence on destruction
$timer->{'silence'} = $opts{'silence'};
# note beginning of duration
if ($opts{'init_title'}) {
if (defined $timer->{'title'}) {
Debug::ShowStuff::println('start timer - ', $timer->{'title'});
}
else {
Debug::ShowStuff::println('start timer');
}
}
# return
return $timer;
}
#
# new
#---------------------------------------------------------------------------
#---------------------------------------------------------------------------
# elapsed
#
sub elapsed {
my ($timer) = @_;
return time() - $timer->{'start'};
}
#
# elapsed
#---------------------------------------------------------------------------
#---------------------------------------------------------------------------
# message
#
sub message {
my ($timer) = @_;
my ($duration, $seconds, $rv);
# get duration
$duration = time() - $timer->{'start'};
# second or seconds
if ($duration == 1)
{ $seconds = 'second' }
else
{ $seconds = 'seconds' }
# note beginning of duration
if (defined $timer->{'title'}) {
$rv =
'duration - ' .
$timer->{'title'} .
': ' .
$duration .
' seconds';
}
else {
$rv =
'duration : ' .
$duration .
' ' .
$seconds;
}
# return
return $rv;
}
#
# message
#---------------------------------------------------------------------------
#---------------------------------------------------------------------------
# silence
#
sub silence {
my ($timer) = @_;
$timer->{'silence'} = 1;
}
#
# silence
#---------------------------------------------------------------------------
#------------------------------------------------------------------------------
# DESTROY
#
DESTROY {
my ($timer) = @_;
# only output info if the 'silence' property is false
if (! $timer->{'silence'}) {
Debug::ShowStuff::println($timer->message());
}
}
#
# DESTROY
#------------------------------------------------------------------------------
#
# Timer
###############################################################################
###############################################################################
# TempShowStuff
#
package Debug::ShowStuff::TempShowStuff;
use strict;
#------------------------------------------------------------------------------
# new
#
sub new {
my ($class, $set) = @_;
my $self = bless {}, $class;
# get current state
$self->{'org'} = $Debug::ShowStuff::active;
# turn showstuff off
Debug::ShowStuff::showstuff($set);
# return
return $self;
}
#
# new
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# DESTROY
#
sub DESTROY {
my ($self) = @_;
# turn showstuff off
Debug::ShowStuff::showstuff($self->{'org'});
}
#
# DESTROY
#------------------------------------------------------------------------------
#
# TempShowStuff
###############################################################################
###############################################################################
# Debug::ShowStuff::SeparatePrint
#
package Debug::ShowStuff::SeparatePrint;
use strict;
use IO::Handle;
use IO::Seekable;
use Symbol;
use Carp 'croak';
#--------------------------------------------------------------------------------------
# new
#
sub new {
my($class, $path, %opts) = @_;
my $fh = gensym;
# must have path
if (! defined $path)
{ croak "Debug::ShowStuff::SeparatePrint->new must have file path" }
# tie handle
${*$fh} = tie *$fh, "${class}::Tie", $path;
bless $fh, $class;
# if "new" parameter is sent and is true, wipe out any existing contents
# in the output path
if ($opts{'new'}) {
FileHandle->new("> $path") or die $!;
}
# return
return $fh;
}
#
# new
#--------------------------------------------------------------------------------------
#
# Debug::ShowStuff::SeparatePrint
###############################################################################
###############################################################################
# Debug::ShowStuff::SeparatePrint::Tie
#
package Debug::ShowStuff::SeparatePrint::Tie;
use strict;
use IO::Seekable;
use FileHandle;
use Fcntl ':mode', ':flock';
#------------------------------------------------------------------------------
# TIEHANDLE
#
sub TIEHANDLE {
my($class, $path) = @_;
my $handle = bless({}, $class);
# hold on to path
$handle->{'path'} = $path;
# return
return $handle;
}
#
# TIEHANDLE
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# don't use these methods
# This type of file handle is for printing only, so none of the following
# methods make sense to use and should notify the programming as such.
#
sub READLINE { die "FileHandle::Separate::Tie is for printing only" }
sub READ { die "FileHandle::Separate::Tie is for printing only" }
sub GETC { die "FileHandle::Separate::Tie is for printing only" }
sub WRITE { die "FileHandle::Separate::Tie is for printing only" }
sub TELL { die "FileHandle::Separate::Tie is for printing only" }
sub SEEK { die "FileHandle::Separate::Tie is for printing only" }
#
# don't use these methods
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# these methods have no effect, but aren't harmful to use
#
sub FLOCK {}
sub BINMODE {}
sub CLOSE {}
#
# these methods have no effect, but aren't harmful to use
#------------------------------------------------------------------------------
#------------------------------------------------------------------------------
# PRINT
#
sub PRINT {
my ($handle, @data) = @_;
my ($out);
# open file handle
$out = FileHandle->new(">> $handle->{'path'}") or
die "unable to open filehandle: $!";
# get exclusive lock
flock($out, LOCK_EX) or
die "unable to lock file: $!";
# seek end of file
$out->seek(0, SEEK_END);
# output
print $out @data;
}
#
# PRINT
#------------------------------------------------------------------------------
#------------------------------------------------
# PRINTF
#
sub PRINTF {
my $self = shift;
return $self->PRINT(sprintf( shift, @_ ));
}
#
# PRINTF
#------------------------------------------------
#
# Debug::ShowStuff::SeparatePrint::Tie
###############################################################################
# return true
1;
__END__
=head1 TERMS AND CONDITIONS
Copyright (c) 2010-2013 by Miko O'Sullivan. All rights reserved. This
program is free software; you can redistribute it and/or modify it under the
same terms as Perl itself. This software comes with B of any kind.
=head1 AUTHORS
Miko O'Sullivan
F
=head1 VERSION
=over
=item Version 1.00 May 29, 2003
Initial public release.
=item Version 1.01 May 29, 2003
Setting sort order of hash keys to case insensitive.
=item Version 1.10 Nov 6, 2010
After seven years, decided to update the version on CPAN.
=item Version 1.11 Nov 13, 2010
Fixed prerequisite requirement for MemHandle and Taint. Added timer()
functions. Some minor documentation fixes and tidying up.
=item Version 1.12 Nov 29, 2010
Changing from using Taint module, which has had a lot of problems, to
Scalar::Util, which is more (but not completely) stable.
=item Version 1.13 Dec 1, 2010
Fixed bug in prerequisites for Scalar::Util.
=item Version 1.14 February 23, 2013
Added showsth, showsql, and explainsql. Added the separateprint option to
setoutput. Tidied up documentation. Fixed problems with prerequisites.
Probably added many other features since the last time I uploaded this module,
but can't remember tham all.
=back
=cut