1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
|
use 5.008001;
use strict;
use warnings;
package Log::Any::Proxy::WithStackTrace;
# ABSTRACT: Log::Any proxy to upgrade string errors to objects with stack traces
our $VERSION = '1.718';
use Log::Any::Proxy;
our @ISA = qw/Log::Any::Proxy/;
use Devel::StackTrace 2.00;
use Log::Any::Adapter::Util ();
use Scalar::Util qw/blessed reftype/;
use overload;
#pod =head1 SYNOPSIS
#pod
#pod use Log::Any qw( $log, proxy_class => 'WithStackTrace' );
#pod
#pod # Allow stack trace call stack arguments to be logged:
#pod use Log::Any qw( $log, proxy_class => 'WithStackTrace',
#pod proxy_show_stack_trace_args => 1 );
#pod
#pod # Configure some adapter that knows how to:
#pod # 1) handle structured data, and
#pod # 2) handle message objects which have a "stack_trace" method:
#pod Log::Any::Adapter->set($adapter);
#pod
#pod $log->error("Help!"); # stack trace gets automatically added,
#pod # starting from this line of code
#pod
#pod =head1 DESCRIPTION
#pod
#pod Some log adapters, like L<Log::Any::Adapter::Sentry::Raven>, are able to
#pod take advantage of being passed message objects that contain a stack
#pod trace. However if a stack trace is not available, and fallback logic is
#pod used to generate one, the resulting trace can be confusing if it begins
#pod relative to where the log adapter was called, and not relative to where
#pod the logging method was originally called.
#pod
#pod With this proxy in place, if any logging method is called with a log
#pod message that is a non-reference scalar (i.e. a string), that log message
#pod will be upgraded into a C<Log::Any::MessageWithStackTrace> object with a
#pod C<stack_trace> method, and that method will return a trace relative to
#pod where the logging method was called. A string overload is provided on
#pod the object to return the original log message.
#pod
#pod Additionally, any call stack arguments in the stack trace will be
#pod deleted before logging, to avoid accidentally logging sensitive data.
#pod This happens both for message objects that were auto-generated from
#pod string messages, as well as for message objects that were passed in
#pod directly (if they appear to have a stack trace method). This default
#pod argument scrubbing behavior can be turned off by specifying a true value
#pod for the C<proxy_show_stack_trace_args> import flag.
#pod
#pod B<Important:> This proxy should be used with a L<Log::Any::Adapter> that
#pod is configured to handle structured data. Otherwise the object created
#pod here will just get stringified before it can be used to access the stack
#pod trace.
#pod
#pod =cut
{
package # hide from PAUSE indexer
Log::Any::MessageWithStackTrace;
use overload '""' => \&stringify;
sub new
{
my ($class, $message, %opts) = @_;
return bless {
message => $message,
stack_trace => Devel::StackTrace->new(
# Filter e.g "Log::Any::Proxy", "My::Log::Any::Proxy", etc.
ignore_package => [ qr/(?:^|::)Log::Any(?:::|$)/ ],
no_args => $opts{no_args},
),
}, $class;
}
sub stringify { $_[0]->{message} }
sub stack_trace { $_[0]->{stack_trace} }
}
#pod =head1 METHODS
#pod
#pod =head2 maybe_upgrade_with_stack_trace
#pod
#pod @args = $self->maybe_upgrade_with_stack_trace(@args);
#pod
#pod This is an internal-use method that will convert a non-reference scalar
#pod message into a C<Log::Any::MessageWithStackTrace> object with a
#pod C<stack_trace> method. A string overload is provided to return the
#pod original message.
#pod
#pod Stack trace args are scrubbed out in case they contain sensitive data,
#pod unless the C<proxy_show_stack_trace_args> option has been set.
#pod
#pod =cut
sub maybe_upgrade_with_stack_trace
{
my ($self, @args) = @_;
# We expect a message, optionally followed by a structured data
# context hashref. Bail if we get anything other than that rather
# than guess what the caller might be trying to do:
return @args unless @args == 1 ||
( @args == 2 && ref $args[1] eq 'HASH' );
if (ref $args[0]) {
$self->maybe_delete_stack_trace_args($args[0])
unless $self->{proxy_show_stack_trace_args};
}
else {
$args[0] = Log::Any::MessageWithStackTrace->new(
$args[0],
no_args => !$self->{proxy_show_stack_trace_args},
);
}
return @args;
}
#pod =head2 maybe_delete_stack_trace_args
#pod
#pod $self->maybe_delete_stack_trace_args($arg);
#pod
#pod This is an internal-use method that, given a single argument that is a
#pod reference, tries to figure out whether the argument is an object with a
#pod stack trace, and if so tries to delete any stack trace args.
#pod
#pod The logic is based on L<Devel::StackTrace::Extract>.
#pod
#pod It specifically looks for objects with a C<stack_trace> method (which
#pod should catch anything that does L<StackTrace::Auto>, including anything
#pod that does L<Throwable::Error>), or a C<trace> method (used by
#pod L<Exception::Class> and L<Moose::Exception> and friends).
#pod
#pod It specifically ignores L<Mojo::Exception> objects, because their stack
#pod traces don't contain any call stack args.
#pod
#pod =cut
sub maybe_delete_stack_trace_args
{
my ($self, $arg) = @_;
return unless blessed $arg;
if ($arg->can('stack_trace')) {
# This should catch anything that does StackTrace::Auto,
# including anything that does Throwable::Error.
my $trace = $arg->stack_trace;
$self->delete_args_from_stack_trace($trace);
}
elsif ($arg->isa('Mojo::Exception')) {
# Skip these, they don't have args in their stack traces.
}
elsif ($arg->can('trace')) {
# This should catch Exception::Class and Moose::Exception and
# friends. Make sure to check for the "trace" method *after*
# skipping the Mojo::Exception objects, because those also have
# a "trace" method.
my $trace = $arg->trace;
$self->delete_args_from_stack_trace($trace);
}
return;
}
my %aliases = Log::Any::Adapter::Util::log_level_aliases();
# Set up methods/aliases and detection methods/aliases
foreach my $name ( Log::Any::Adapter::Util::logging_methods(), keys(%aliases) )
{
my $super_name = "SUPER::" . $name;
no strict 'refs';
*{$name} = sub {
my ($self, @args) = @_;
@args = $self->maybe_upgrade_with_stack_trace(@args);
my $response = $self->$super_name(@args);
return $response if defined wantarray;
return;
};
}
#pod =head2 delete_args_from_stack_trace($trace)
#pod
#pod $self->delete_args_from_stack_trace($trace)
#pod
#pod To scrub potentially sensitive data from C<Devel::StackTrace> arguments,
#pod this method deletes arguments from all of the C<Devel::StackTrace::Frame>
#pod in the trace.
#pod
#pod =cut
sub delete_args_from_stack_trace
{
my ($self, $trace) = @_;
return unless $trace && $trace->can('frames');
foreach my $frame ($trace->frames) {
next unless $frame->{args};
$frame->{args} = [];
}
return;
}
1;
__END__
=pod
=encoding UTF-8
=head1 NAME
Log::Any::Proxy::WithStackTrace - Log::Any proxy to upgrade string errors to objects with stack traces
=head1 VERSION
version 1.718
=head1 SYNOPSIS
use Log::Any qw( $log, proxy_class => 'WithStackTrace' );
# Allow stack trace call stack arguments to be logged:
use Log::Any qw( $log, proxy_class => 'WithStackTrace',
proxy_show_stack_trace_args => 1 );
# Configure some adapter that knows how to:
# 1) handle structured data, and
# 2) handle message objects which have a "stack_trace" method:
Log::Any::Adapter->set($adapter);
$log->error("Help!"); # stack trace gets automatically added,
# starting from this line of code
=head1 DESCRIPTION
Some log adapters, like L<Log::Any::Adapter::Sentry::Raven>, are able to
take advantage of being passed message objects that contain a stack
trace. However if a stack trace is not available, and fallback logic is
used to generate one, the resulting trace can be confusing if it begins
relative to where the log adapter was called, and not relative to where
the logging method was originally called.
With this proxy in place, if any logging method is called with a log
message that is a non-reference scalar (i.e. a string), that log message
will be upgraded into a C<Log::Any::MessageWithStackTrace> object with a
C<stack_trace> method, and that method will return a trace relative to
where the logging method was called. A string overload is provided on
the object to return the original log message.
Additionally, any call stack arguments in the stack trace will be
deleted before logging, to avoid accidentally logging sensitive data.
This happens both for message objects that were auto-generated from
string messages, as well as for message objects that were passed in
directly (if they appear to have a stack trace method). This default
argument scrubbing behavior can be turned off by specifying a true value
for the C<proxy_show_stack_trace_args> import flag.
B<Important:> This proxy should be used with a L<Log::Any::Adapter> that
is configured to handle structured data. Otherwise the object created
here will just get stringified before it can be used to access the stack
trace.
=head1 METHODS
=head2 maybe_upgrade_with_stack_trace
@args = $self->maybe_upgrade_with_stack_trace(@args);
This is an internal-use method that will convert a non-reference scalar
message into a C<Log::Any::MessageWithStackTrace> object with a
C<stack_trace> method. A string overload is provided to return the
original message.
Stack trace args are scrubbed out in case they contain sensitive data,
unless the C<proxy_show_stack_trace_args> option has been set.
=head2 maybe_delete_stack_trace_args
$self->maybe_delete_stack_trace_args($arg);
This is an internal-use method that, given a single argument that is a
reference, tries to figure out whether the argument is an object with a
stack trace, and if so tries to delete any stack trace args.
The logic is based on L<Devel::StackTrace::Extract>.
It specifically looks for objects with a C<stack_trace> method (which
should catch anything that does L<StackTrace::Auto>, including anything
that does L<Throwable::Error>), or a C<trace> method (used by
L<Exception::Class> and L<Moose::Exception> and friends).
It specifically ignores L<Mojo::Exception> objects, because their stack
traces don't contain any call stack args.
=head2 delete_args_from_stack_trace($trace)
$self->delete_args_from_stack_trace($trace)
To scrub potentially sensitive data from C<Devel::StackTrace> arguments,
this method deletes arguments from all of the C<Devel::StackTrace::Frame>
in the trace.
=head1 AUTHORS
=over 4
=item *
Jonathan Swartz <swartz@pobox.com>
=item *
David Golden <dagolden@cpan.org>
=item *
Doug Bell <preaction@cpan.org>
=item *
Daniel Pittman <daniel@rimspace.net>
=item *
Stephen Thirlwall <sdt@cpan.org>
=back
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2017 by Jonathan Swartz, David Golden, and Doug Bell.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut
|