176 lines
5.5 KiB
Perl
176 lines
5.5 KiB
Perl
|
# ====================================================================
|
||
|
# Gossamer Threads Module Library - http://gossamer-threads.com/
|
||
|
#
|
||
|
# GT::SQL::Driver::debug
|
||
|
# Author: Jason Rhinelander
|
||
|
# CVS Info :
|
||
|
# $Id: debug.pm,v 2.0 2004/08/28 03:51:31 jagerman Exp $
|
||
|
#
|
||
|
# Copyright (c) 2004 Gossamer Threads Inc. All Rights Reserved.
|
||
|
# ====================================================================
|
||
|
#
|
||
|
# Description:
|
||
|
# GT::SQL::Driver debugging module
|
||
|
#
|
||
|
|
||
|
package GT::SQL::Driver::debug;
|
||
|
use strict;
|
||
|
|
||
|
use strict;
|
||
|
use GT::AutoLoader;
|
||
|
use vars qw/$LAST_QUERY @QUERY_STACK @STACK_TRACE $QUERY_STACK_SIZE @ISA/;
|
||
|
@ISA = qw(GT::Base);
|
||
|
$QUERY_STACK_SIZE = 100;
|
||
|
|
||
|
$COMPILE{last_query} = __LINE__ . <<'END_OF_SUB';
|
||
|
sub last_query {
|
||
|
# -------------------------------------------------------------------
|
||
|
# Get, or set the last query.
|
||
|
#
|
||
|
my $self = shift;
|
||
|
return $self->error('NEEDDEBUG', 'WARN') if (! $self->{_debug});
|
||
|
|
||
|
@_ > 0 or return $LAST_QUERY || '';
|
||
|
|
||
|
$LAST_QUERY = shift;
|
||
|
$LAST_QUERY = GT::SQL::Driver::debug->replace_placeholders($LAST_QUERY, @_) if (@_);
|
||
|
|
||
|
# Display stack traces if requested via debug level.
|
||
|
my $stack = '';
|
||
|
if ($self->{_debug} > 2) {
|
||
|
($stack, $LAST_QUERY) = js_stack(3, $LAST_QUERY);
|
||
|
}
|
||
|
elsif ($self->{_debug} > 1) {
|
||
|
package DB;
|
||
|
my $i = 2;
|
||
|
my $ls = defined $ENV{REQUEST_METHOD} ? '<br>' : "\n";
|
||
|
my $spc = defined $ENV{REQUEST_METHOD} ? ' ' : ' ';
|
||
|
while (my ($file, $line, $sub, $args) = (caller($i++))[1,2,3,4]) {
|
||
|
my @args;
|
||
|
for (@DB::args) {
|
||
|
eval { my $a = $_ }; # workaround for a reference that doesn't think it's a reference
|
||
|
my $print = $@ ? \$_ : $_;
|
||
|
push @args, defined $print ? $print : '[undef]';
|
||
|
}
|
||
|
if (@args) {
|
||
|
my $args = join ", ", @args;
|
||
|
$args =~ s/\n\s*\n/\n/g;
|
||
|
$args =~ s/\n/\n$spc$spc$spc$spc/g;
|
||
|
$stack .= qq!$sub called at $file line $line with arguments $ls$spc$spc ($args).$ls!;
|
||
|
}
|
||
|
else {
|
||
|
$stack .= qq!$sub called at $file line $line with no arguments.$ls!;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
push @QUERY_STACK, $LAST_QUERY;
|
||
|
push @STACK_TRACE, "<blockquote>\n" . $stack . "\n</blockquote>\n" if ($self->{_debug} and $stack);
|
||
|
|
||
|
# Pesistance such as Mod_Perl
|
||
|
@QUERY_STACK > $QUERY_STACK_SIZE and shift @QUERY_STACK;
|
||
|
@STACK_TRACE > $QUERY_STACK_SIZE and shift @STACK_TRACE;
|
||
|
|
||
|
return $LAST_QUERY || '';
|
||
|
}
|
||
|
END_OF_SUB
|
||
|
|
||
|
$COMPILE{js_stack} = __LINE__ . <<'END_OF_SUB';
|
||
|
sub js_stack {
|
||
|
# -------------------------------------------------------------------
|
||
|
# Create a nicely formatted javascript browser that (unfortunately)
|
||
|
# only works in ie, netscape sucks.
|
||
|
#
|
||
|
my ($sp, $title) = @_;
|
||
|
|
||
|
my $nb = @QUERY_STACK;
|
||
|
my ($stack, $dump_out);
|
||
|
{
|
||
|
package DB;
|
||
|
require GT::Dumper;
|
||
|
my $i = 0;
|
||
|
|
||
|
while (my ($file, $line, $sub, $args) = (caller($sp++))[1,2,3,4]) {
|
||
|
if (@DB::args) {
|
||
|
$args = "with arguments<br> ";
|
||
|
my @args;
|
||
|
for (@DB::args) {
|
||
|
eval { my $a = $_ }; # workaround for a reference that doesn't think it's a reference
|
||
|
my $print = $@ ? \$_ : $_;
|
||
|
my $arg = defined $print ? $print : '[undef]';
|
||
|
|
||
|
$args .= "<a href='#a$nb$i'>$arg</a>, ";
|
||
|
my $dump = GT::Dumper::Dumper($arg);
|
||
|
$dump_out .= qq~
|
||
|
<a name="a$nb$i"></a>
|
||
|
<a href="#top">Top</a>
|
||
|
<pre>$dump</pre>
|
||
|
~;
|
||
|
$i++;
|
||
|
}
|
||
|
chop $args; chop $args;
|
||
|
}
|
||
|
else {
|
||
|
$args = "with no arguments";
|
||
|
}
|
||
|
$stack .= qq!<li>$sub called at $file line $line $args.<br></li>\n!;
|
||
|
}
|
||
|
}
|
||
|
$stack =~ s/\\/\\\\/g;
|
||
|
$stack =~ s/[\n\r]+/\\n/g;
|
||
|
$stack =~ s/'/\\'/g;
|
||
|
$stack =~ s,script,sc'+'ript,g;
|
||
|
|
||
|
$dump_out =~ s/\\/\\\\/g;
|
||
|
$dump_out =~ s/[\n\r]+/\\n/g;
|
||
|
|
||
|
$dump_out =~ s/'/\\'/g;
|
||
|
$dump_out =~ s,script,sc'+'ript,g;
|
||
|
|
||
|
my $var = <<HTML;
|
||
|
<script language="JavaScript">
|
||
|
function my$nb () {
|
||
|
msg = window.open('','my$nb','resizable=yes,width=700,height=500,scrollbars=yes');
|
||
|
msg.document.write('<html><body><a name="top"></a>STACK TRACE<BR><OL>$stack</OL>$dump_out</BODY></HTML>');
|
||
|
msg.document.close();
|
||
|
}
|
||
|
HTML
|
||
|
my $link = qq!<a href="javascript:my$nb();">$title</a><br>!;
|
||
|
|
||
|
return $var, $link;
|
||
|
}
|
||
|
END_OF_SUB
|
||
|
|
||
|
$COMPILE{quick_quote} = __LINE__ . <<'END_OF_SUB';
|
||
|
sub quick_quote {
|
||
|
# -------------------------------------------------------------------
|
||
|
# Quick quote to replace ' with \'.
|
||
|
#
|
||
|
my $str = shift;
|
||
|
defined $str and ($str eq "") and return "''";
|
||
|
$str =~ s/'/\\'/g;
|
||
|
return $str;
|
||
|
}
|
||
|
END_OF_SUB
|
||
|
|
||
|
$COMPILE{replace_placeholders} = __LINE__ . <<'END_OF_SUB';
|
||
|
sub replace_placeholders {
|
||
|
# -------------------------------------------------------------------
|
||
|
# Replace question marks with the actual values
|
||
|
#
|
||
|
my ($self, $query, @args) = @_;
|
||
|
if (@args > 0) {
|
||
|
my @vals = split /('(?:[^']+|''|\\')')/, $query;
|
||
|
VALUE: for my $val (@args) {
|
||
|
SUBSTRING: for my $i (0 .. $#vals) {
|
||
|
next SUBSTRING if $i % 2;
|
||
|
next VALUE if $vals[$i] =~ s/\?/defined $val ? ( $val =~ m,\D, ? "'".quick_quote($val)."'" : quick_quote($val) ) : 'NULL'/e;
|
||
|
}
|
||
|
}
|
||
|
$query = join '', @vals;
|
||
|
}
|
||
|
return $query;
|
||
|
}
|
||
|
END_OF_SUB
|
||
|
|
||
|
1;
|