# @(#)$Id: Exception.pm 1181 2012-04-17 19:06:07Z pjf $ package CatalystX::Usul::Exception; use strict; use warnings; use version; our $VERSION = qv( sprintf '0.7.%d', q$Rev: 1181 $ =~ /\d+/gmx ); use Exception::Class 'CatalystX::Usul::Exception::Base' => { fields => [ qw(args leader out rv) ] }; use base qw(CatalystX::Usul::Exception::Base); use Carp; use MRO::Compat; use CatalystX::Usul::Constants; use English qw(-no_match_vars); use Scalar::Util qw(blessed); our $IGNORE = [ __PACKAGE__, qw(CatalystX::Usul::IPC) ]; sub new { my ($self, @args) = @_; my $args = __arg_list( @args ); my $level = 3; exists $args->{level} and $level = delete $args->{level}; my ($package, $line) = (caller( $level ))[ 0, 2 ]; my $leader = "${package}[${line}]: "; if (__is_one_of_us( $args->{error} )) { $args->{error}->{leader} = $leader; return $args->{error}; } $args->{error} .= NUL; return $self->next::method( args => [], error => 'Error unknown', ignore_package => $IGNORE, leader => $leader, out => NUL, rv => 1, %{ $args } ); } sub catch { my ($self, @args) = @_; my $args = __arg_list( @args ); $args->{error} ||= $EVAL_ERROR; $args->{error} or return; return __is_one_of_us( $args->{error} ) ? $args->{error} : $self->new( $args ); } sub full_message { my $self = shift; my $text = $self->error or return; # Expand positional parameters of the form [_] 0 > index $text, LOCALIZE and return $self->leader.$text; my @args = @{ $self->args }; push @args, map { NUL } 0 .. 10; $text =~ s{ \[ _ (\d+) \] }{$args[ $1 - 1 ]}gmx; return $self->leader.$text; } sub stacktrace { my ($self, $skip) = @_; my ($l_no, @lines, %seen, $subr); for my $frame (reverse $self->trace->frames) { unless ($l_no = $seen{ $frame->package } and $l_no == $frame->line) { $subr and push @lines, join SPC, $subr, 'line', $frame->line; $seen{ $frame->package } = $frame->line; } $subr = $frame->subroutine; } defined $skip or $skip = 1; pop @lines while ($skip--); return wantarray ? reverse @lines : (join "\n", reverse @lines)."\n"; } sub throw { my ($self, @rest) = @_; croak __is_one_of_us( $rest[ 0 ] ) ? $rest[ 0 ] : $self->new( @rest ); } sub throw_on_error { my ($self, @rest) = @_; my $e; $e = $self->catch( @rest ) and $self->throw( $e ); return; } # Private subroutines sub __arg_list { return $_[ 0 ] && ref $_[ 0 ] eq HASH ? { %{ $_[ 0 ] } } : $_[ 0 ] && defined $_[ 1 ] ? { @_ } : { error => $_[ 0 ] }; } sub __is_one_of_us { return $_[ 0 ] && blessed $_[ 0 ] && $_[ 0 ]->isa( __PACKAGE__ ); } 1; __END__ =pod =head1 Name CatalystX::Usul::Exception - Exception base class =head1 Version 0.7.$Revision: 1181 $ =head1 Synopsis use base qw(CatalystX::Usul); use Try::Tiny; sub some_method { my $self = shift; eval { this_will_fail }; $self->throw_on_error; OR try { this_will_fail } catch { $self->throw( $_ ) }; } =head1 Description Implements try (by way of an eval), throw, and catch error semantics. Inherits from L =head1 Subroutines/Methods =head2 new Create an exception object. You probably do not want to call this directly, but indirectly through L and L Calls the L method if asked to serialize =head2 catch Catches and returns a thrown exception or generates a new exception if I has been set. Returns either an exception object or undef =head2 full_message This is what the object stringifies to =head2 stacktrace $lines = $e->stacktrace( $num_lines_to_skip ); Return the stack trace. Defaults to skipping one (the first) line of output =head2 throw Create (or re-throw) an exception to be caught by the catch above. If the passed parameter is a blessed reference it is re-thrown. If a single scalar is passed it is taken to be an error message code, a new exception is created with all other parameters taking their default values. If more than one parameter is passed the it is treated as a list and used to instantiate the new exception. The 'error' parameter must be provided in this case =head2 throw_on_error Calls L and if the was an exception Ls it =head1 Diagnostics None =head1 Configuration and Environment The C<$IGNORE> package variable is list of methods whose presence should be suppressed in the stack trace output =head1 Dependencies =over 3 =item L =item L =item L =back =head1 Incompatibilities There are no known incompatibilities in this module =head1 Bugs and Limitations The default ignore package list should be configurable There are no known bugs in this module. Please report problems to the address below. Patches are welcome =head1 Author Peter Flanigan C<< >> =head1 License and Copyright Copyright (c) 2008 Peter Flanigan. All rights reserved This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. See L This program is distributed in the hope that it will be useful, but WITHOUT WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE =cut # Local Variables: # mode: perl # tab-width: 3 # End: