1 package File
::KDBX
::Error
;
2 # ABSTRACT: Represents something bad that happened
8 use Exporter
qw(import);
9 use Scalar
::Util
qw(blessed looks_like_number);
10 use namespace
::clean
-except
=> 'import';
12 our $VERSION = '999.999'; # VERSION
14 our @EXPORT = qw(alert error throw);
16 my $WARNINGS_CATEGORY;
18 $WARNINGS_CATEGORY = 'File::KDBX';
19 if (warnings-
>can('register_categories')) {
20 warnings
::register_categories
($WARNINGS_CATEGORY);
23 eval qq{package $WARNINGS_CATEGORY; use warnings::register; 1}; ## no critic ProhibitStringyEval
26 my $debug = $ENV{DEBUG
};
27 $debug = looks_like_number
($debug) ? (0 + $debug) : ($debug ? 1 : 0);
28 *_DEBUG
= $debug == 1 ? sub() { 1 } :
29 $debug == 2 ? sub() { 2 } :
30 $debug == 3 ? sub() { 3 } :
31 $debug == 4 ? sub() { 4 } : sub() { 0 };
34 use overload
'""' => 'to_string', cmp => '_cmp';
38 $error = File
::KDBX
::Error-
>new($message, %details);
40 Construct a new error
.
46 my %args = @_ % 2 == 0 ? @_ : (_error
=> shift, @_);
48 my $error = delete $args{_error
};
50 $e =~ s/ at \H+ line \d+.*//g;
54 error
=> $e // 'Something happened',
59 local $Carp::CarpInternal
{''.__PACKAGE__
} = 1;
60 my $mess = $error =~ /at \H+ line \d+/ ? $error : Carp
::longmess
($error);
61 [map { /^\h*(.*?)\.?$/ ? $1 : $_ } split(/\n/, $mess)];
70 $error = error
($error);
71 $error = error
($message, %details);
72 $error = File
::KDBX
::Error-
>error($error);
73 $error = File
::KDBX
::Error-
>error($message, %details);
75 Wrap a thing to make it an error object
. If the thing
is already an error
, it gets returned
. Otherwise what
is
76 passed will be forwarded to L
</new
> to create a new error object
.
78 This can be convenient
for error handling
when you
're not sure what the exception is but you want to treat it
79 as a B<File::KDBX::Error>. Example:
82 if (my $error = error(@_)) {
83 if ($error->type eq 'key
.missing
') {
84 handle_missing_key($error);
87 handle_other_error($error);
94 my $class = @_ && $_[0] eq __PACKAGE__ ? shift : undef;
95 my $self = (blessed($_[0]) && $_[0]->isa('File
::KDBX
::Error
'))
99 : __PACKAGE__->new(@_);
105 \%details = $error->details;
107 Get the error details.
114 my $details = $self->{details} //= {};
115 @$details{keys %args} = values %args;
121 Get the value of C<errno> when the exception was created.
125 Get the value of C<$@> (i.e. latest exception) at the time the exception was created.
129 Get a stack trace indicating where in the code the exception was created.
135 Get the exception type, if any.
139 sub errno { $_[0]->{errno} }
140 sub previous { $_[0]->{previous} }
141 sub trace { $_[0]->{trace} // [] }
142 sub type { $_[0]->details->{type} // '' }
146 $message = $error->to_string;
151 This does not contain a stack trace, but you can set the C<DEBUG> environment variable to at least 2 to
152 stringify the whole error object.
156 sub _cmp { "$_[0]" cmp "$_[1]" }
160 my $msg = "$self->{trace}[0]";
161 $msg .= '.' if $msg !~ /[\.\!\?]$/;
163 require Data::Dumper;
164 local $Data::Dumper::Indent = 1;
165 local $Data::Dumper::Quotekeys = 0;
166 local $Data::Dumper::Sortkeys = 1;
167 local $Data::Dumper::Terse = 1;
168 local $Data::Dumper::Trailingcomma = 1;
169 local $Data::Dumper::Useqq = 1;
170 $msg .= "\n" . Data::Dumper::Dumper $self;
172 $msg .= "\n" if $msg !~ /\n$/;
178 File::KDBX::Error::throw($message, %details);
186 my $self = error(@_);
192 File::KDBX::Error::warn($message, %details);
200 return if !($File::KDBX::WARNINGS // 1);
202 my $self = error(@_);
204 # Use die and warn directly instead of warnings::warnif because the latter only provides the stringified
205 # error to the warning signal handler (perl 5.34). Maybe that's a warnings
.pm bug
?
207 if (my $fatal = warnings-
>can('fatal_enabled_at_level')) {
208 my $blame = _find_blame_frame
();
209 die $self if $fatal->($WARNINGS_CATEGORY, $blame);
212 if (my $enabled = warnings-
>can('enabled_at_level')) {
213 my $blame = _find_blame_frame
();
214 warn $self if $enabled->($WARNINGS_CATEGORY, $blame);
216 elsif ($enabled = warnings-
>can('enabled')) {
217 warn $self if $enabled->($WARNINGS_CATEGORY);
229 Importable alias
for L
</warn>.
233 sub alert
{ goto &warn }
235 sub _find_blame_frame
{
238 my ($package) = caller($frame);
240 return $frame - 1 if $package !~ /^\Q$WARNINGS_CATEGORY\E/;