Document $SIG{__DIE__} behavior (or lack thereof)
[p5sagit/Try-Tiny.git] / lib / Try / Tiny.pm
CommitLineData
3176feef 1package Try::Tiny;
2
3use strict;
ae53da51 4#use warnings;
3176feef 5
ae53da51 6use vars qw(@EXPORT @EXPORT_OK $VERSION @ISA);
7
8BEGIN {
9 require Exporter;
10 @ISA = qw(Exporter);
11}
3176feef 12
aa03b5ce 13$VERSION = "0.04";
3176feef 14
15$VERSION = eval $VERSION;
16
7195fc08 17@EXPORT = @EXPORT_OK = qw(try catch finally);
3176feef 18
6f114080 19$Carp::Internal{+__PACKAGE__}++;
20
7195fc08 21# Need to prototype as @ not $$ because of the way Perl evaluates the prototype.
22# Keeping it at $$ means you only ever get 1 sub because we need to eval in a list
23# context & not a scalar one
24
25sub try (&;@) {
26 my ( $try, @code_refs ) = @_;
3176feef 27
28 # we need to save this here, the eval block will be in scalar context due
29 # to $failed
30 my $wantarray = wantarray;
31
7195fc08 32 my ( $catch, $finally );
33
34 # find labeled blocks in the argument list.
35 # catch and finally tag the blocks by blessing a scalar reference to them.
36 foreach my $code_ref (@code_refs) {
37 next unless $code_ref;
38
39 my $ref = ref($code_ref);
40
41 if ( $ref eq 'Try::Tiny::Catch' ) {
42 $catch = ${$code_ref};
43 } elsif ( $ref eq 'Try::Tiny::Finally' ) {
44 $finally = ${$code_ref};
45 } else {
46 use Carp;
47 confess("Unknown code ref type given '${ref}'. Check your usage & try again");
48 }
49 }
50
1d64c1ad 51 # save the value of $@ so we can set $@ back to it in the beginning of the eval
511c05ca 52 my $prev_error = $@;
53
3176feef 54 my ( @ret, $error, $failed );
55
1d64c1ad 56 # FIXME consider using local $SIG{__DIE__} to accumulate all errors. It's
3176feef 57 # not perfect, but we could provide a list of additional errors for
58 # $catch->();
59
60 {
61 # localize $@ to prevent clobbering of previous value by a successful
62 # eval.
63 local $@;
64
65 # failed will be true if the eval dies, because 1 will not be returned
66 # from the eval body
67 $failed = not eval {
511c05ca 68 $@ = $prev_error;
3176feef 69
70 # evaluate the try block in the correct context
71 if ( $wantarray ) {
72 @ret = $try->();
73 } elsif ( defined $wantarray ) {
74 $ret[0] = $try->();
75 } else {
76 $try->();
77 };
78
79 return 1; # properly set $fail to false
80 };
81
1d64c1ad 82 # copy $@ to $error; when we leave this scope, local $@ will revert $@
3176feef 83 # back to its previous value
84 $error = $@;
85 }
86
82ef0e61 87 # set up a scope guard to invoke the finally block at the end
88 my $guard = $finally && bless \$finally, "Try::Tiny::ScopeGuard";
89
1d64c1ad 90 # at this point $failed contains a true value if the eval died, even if some
91 # destructor overwrote $@ as the eval was unwinding.
3176feef 92 if ( $failed ) {
93 # if we got an error, invoke the catch block.
94 if ( $catch ) {
95 # This works like given($error), but is backwards compatible and
96 # sets $_ in the dynamic scope for the body of C<$catch>
97 for ($error) {
82ef0e61 98 return $catch->($error);
3176feef 99 }
44599111 100
101 # in case when() was used without an explicit return, the C<for>
102 # loop will be aborted and there's no useful return value
3176feef 103 }
44599111 104
105 return;
3176feef 106 } else {
107 # no failure, $@ is back to what it was, everything is fine
108 return $wantarray ? @ret : $ret[0];
109 }
110}
111
7195fc08 112sub catch (&;@) {
113 my ( $block, @rest ) = @_;
114
115 return (
116 bless(\$block, 'Try::Tiny::Catch'),
117 @rest,
118 );
3176feef 119}
120
7195fc08 121sub finally (&;@) {
122 my ( $block, @rest ) = @_;
123
124 return (
125 bless(\$block, 'Try::Tiny::Finally'),
126 @rest,
127 );
128}
3176feef 129
82ef0e61 130sub Try::Tiny::ScopeGuard::DESTROY {
131 my $self = shift;
132 $$self->();
133}
134
3176feef 135__PACKAGE__
136
137__END__
138
139=pod
140
141=head1 NAME
142
143Try::Tiny - minimal try/catch with proper localization of $@
144
145=head1 SYNOPSIS
146
147 # handle errors with a catch handler
148 try {
149 die "foo";
150 } catch {
151 warn "caught error: $_";
152 };
153
154 # just silence errors
155 try {
156 die "foo";
157 };
158
159=head1 DESCRIPTION
160
7195fc08 161This module provides bare bones C<try>/C<catch>/C<finally> statements that are designed to
1f7c5af6 162minimize common mistakes with eval blocks, and NOTHING else.
3176feef 163
164This is unlike L<TryCatch> which provides a nice syntax and avoids adding
165another call stack layer, and supports calling C<return> from the try block to
166return from the parent subroutine. These extra features come at a cost of a few
167dependencies, namely L<Devel::Declare> and L<Scope::Upper> which are
1f7c5af6 168occasionally problematic, and the additional catch filtering uses L<Moose>
169type constraints which may not be desirable either.
3176feef 170
1f7c5af6 171The main focus of this module is to provide simple and reliable error handling
3176feef 172for those having a hard time installing L<TryCatch>, but who still want to
173write correct C<eval> blocks without 5 lines of boilerplate each time.
174
175It's designed to work as correctly as possible in light of the various
176pathological edge cases (see L<BACKGROUND>) and to be compatible with any style
177of error values (simple strings, references, objects, overloaded objects, etc).
178
a5cd5f73 179If the try block dies, it returns the value of the last statement executed in
7195fc08 180the catch block, if there is one. Otherwise, it returns C<undef> in scalar
a5cd5f73 181context or the empty list in list context. The following two examples both
182assign C<"bar"> to C<$x>.
183
184 my $x = try { die "foo" } catch { "bar" };
185
186 my $x = eval { die "foo" } || "bar";
187
7195fc08 188You can add finally blocks making the following true.
189
190 my $x;
191 try { die 'foo' } finally { $x = 'bar' };
192 try { die 'foo' } catch { warn "Got a die: $_" } finally { $x = 'bar' };
193
194Finally blocks are always executed making them suitable for cleanup code
195which cannot be handled using local.
196
3176feef 197=head1 EXPORTS
198
1f7c5af6 199All functions are exported by default using L<Exporter>.
3176feef 200
7195fc08 201If you need to rename the C<try>, C<catch> or C<finally> keyword consider using
6157bcb8 202L<Sub::Import> to get L<Sub::Exporter>'s flexibility.
3176feef 203
204=over 4
205
7195fc08 206=item try (&;@)
3176feef 207
7195fc08 208Takes one mandatory try subroutine, an optional catch subroutine & finally
209subroutine.
3176feef 210
211The mandatory subroutine is evaluated in the context of an C<eval> block.
212
1f7c5af6 213If no error occurred the value from the first block is returned, preserving
214list/scalar context.
3176feef 215
216If there was an error and the second subroutine was given it will be invoked
217with the error in C<$_> (localized) and as that block's first and only
218argument.
219
1f7c5af6 220Note that the error may be false, but if that happens the C<catch> block will
1d64c1ad 221still be invoked.
3176feef 222
7195fc08 223Once all execution is finished then the finally block if given will execute.
224
225=item catch (&;$)
1f7c5af6 226
227Intended to be used in the second argument position of C<try>.
3176feef 228
7195fc08 229Returns a reference to the subroutine it was given but blessed as
230C<Try::Tiny::Catch> which allows try to decode correctly what to do
231with this code reference.
3176feef 232
233 catch { ... }
234
ac4f5f9f 235Inside the catch block the previous value of C<$@> is still available for use.
1d64c1ad 236This value may or may not be meaningful depending on what happened before the
ac4f5f9f 237C<try>, but it might be a good idea to preserve it in an error stack.
238
7195fc08 239=item finally (&;$)
240
241 try { ... }
242 catch { ... }
243 finally { ... };
244
245Or
246
247 try { ... }
248 finally { ... };
249
250Or even
251
252 try { ... }
253 finally { ... }
254 catch { ... };
255
256Intended to be the second or third element of C<try>. Finally blocks are always
257executed in the event of a successful C<try> or if C<catch> is run. This allows
258you to locate cleanup code which cannot be done via C<local()> e.g. closing a file
259handle.
260
261B<You must always do your own error handling in the finally block>. C<Try::Tiny> will
262not do anything about handling possible errors coming from code located in these
263blocks.
264
265In the same way C<catch()> blesses the code reference this subroutine does the same
266except it bless them as C<Try::Tiny::Finally>.
267
3176feef 268=back
269
270=head1 BACKGROUND
271
272There are a number of issues with C<eval>.
273
274=head2 Clobbering $@
275
276When you run an eval block and it succeeds, C<$@> will be cleared, potentially
a717a876 277clobbering an error that is currently being caught.
3176feef 278
1f7c5af6 279This causes action at a distance, clearing previous errors your caller may have
280not yet handled.
281
282C<$@> must be properly localized before invoking C<eval> in order to avoid this
283issue.
3176feef 284
8e5b4441 285More specifically, C<$@> is clobbered at the beginning of the C<eval>, which
511c05ca 286also makes it impossible to capture the previous error before you die (for
287instance when making exception objects with error stacks).
288
289For this reason C<try> will actually set C<$@> to its previous value (before
1d64c1ad 290the localization) in the beginning of the C<eval> block.
511c05ca 291
3176feef 292=head2 Localizing $@ silently masks errors
293
294Inside an eval block C<die> behaves sort of like:
295
296 sub die {
91254b51 297 $@ = $_[0];
3176feef 298 return_undef_from_eval();
299 }
300
301This means that if you were polite and localized C<$@> you can't die in that
1f7c5af6 302scope, or your error will be discarded (printing "Something's wrong" instead).
3176feef 303
304The workaround is very ugly:
305
306 my $error = do {
307 local $@;
308 eval { ... };
309 $@;
310 };
311
312 ...
313 die $error;
314
315=head2 $@ might not be a true value
316
317This code is wrong:
318
319 if ( $@ ) {
320 ...
321 }
322
1f7c5af6 323because due to the previous caveats it may have been unset.
324
1d64c1ad 325C<$@> could also be an overloaded error object that evaluates to false, but
326that's asking for trouble anyway.
3176feef 327
328The classic failure mode is:
329
330 sub Object::DESTROY {
331 eval { ... }
332 }
333
334 eval {
335 my $obj = Object->new;
336
337 die "foo";
338 };
339
340 if ( $@ ) {
341
342 }
343
1f7c5af6 344In this case since C<Object::DESTROY> is not localizing C<$@> but still uses
1d64c1ad 345C<eval>, it will set C<$@> to C<"">.
3176feef 346
1f7c5af6 347The destructor is called when the stack is unwound, after C<die> sets C<$@> to
3176feef 348C<"foo at Foo.pm line 42\n">, so by the time C<if ( $@ )> is evaluated it has
1f7c5af6 349been cleared by C<eval> in the destructor.
3176feef 350
1f7c5af6 351The workaround for this is even uglier than the previous ones. Even though we
352can't save the value of C<$@> from code that doesn't localize, we can at least
353be sure the eval was aborted due to an error:
3176feef 354
355 my $failed = not eval {
356 ...
357
358 return 1;
359 };
360
1f7c5af6 361This is because an C<eval> that caught a C<die> will always return a false
362value.
3176feef 363
f9b91e2c 364=head1 SHINY SYNTAX
3176feef 365
1f7c5af6 366Using Perl 5.10 you can use L<perlsyn/"Switch statements">.
3176feef 367
1f7c5af6 368The C<catch> block is invoked in a topicalizer context (like a C<given> block),
369but note that you can't return a useful value from C<catch> using the C<when>
27293e40 370blocks without an explicit C<return>.
3176feef 371
372This is somewhat similar to Perl 6's C<CATCH> blocks. You can use it to
373concisely match errors:
374
375 try {
376 require Foo;
377 } catch {
1f7c5af6 378 when (/^Can't locate .*?\.pm in \@INC/) { } # ignore
3176feef 379 default { die $_ }
deb85b37 380 };
3176feef 381
382=head1 CAVEATS
383
384=over 4
385
386=item *
387
318cb1eb 388C<@_> is not available, you need to name your args:
389
390 sub foo {
391 my ( $self, @args ) = @_;
392 try { $self->bar(@args) }
393 }
394
395=item *
396
397C<return> returns from the C<try> block, not from the parent sub (note that
398this is also how C<eval> works, but not how L<TryCatch> works):
399
400 sub bar {
401 try { return "foo" };
402 return "baz";
403 }
404
405 say bar(); # "baz"
406
407=item *
408
1f7c5af6 409C<try> introduces another caller stack frame. L<Sub::Uplevel> is not used. L<Carp>
410will report this when using full stack traces. This lack of magic is considered
411a feature.
3176feef 412
413=item *
414
57c50f41 415The value of C<$_> in the C<catch> block is not guaranteed to be the value of
416the exception thrown (C<$@>) in the C<try> block. There is no safe way to
417ensure this, since C<eval> may be used unhygenically in destructors. The only
418guarantee is that the C<catch> will be called if an exception is thrown.
3176feef 419
a5cd5f73 420=item *
421
422The return value of the C<catch> block is not ignored, so if testing the result
423of the expression for truth on success, be sure to return a false value from
424the C<catch> block:
425
426 my $obj = try {
427 MightFail->new;
428 } catch {
429 ...
430
431 return; # avoid returning a true value;
432 };
433
434 return unless $obj;
435
eaca95b7 436=item *
437
438C<$SIG{__DIE__}> is still in effect.
439
440Though it can be argued that C<$SIG{__DIE__}> should be disabled inside of
441C<eval> blocks, since it isn't people have grown to rely on it. Therefore in
442the interests of compatibility, C<try> does not disable C<$SIG{__DIE__}> for
443the scope of the error throwing code.
444
3176feef 445=back
446
447=head1 SEE ALSO
448
449=over 4
450
451=item L<TryCatch>
452
453Much more feature complete, more convenient semantics, but at the cost of
454implementation complexity.
455
9bc603cb 456=item L<autodie>
457
458Automatic error throwing for builtin functions and more. Also designed to
459work well with C<given>/C<when>.
460
f8227e43 461=item L<Throwable>
462
463A lightweight role for rolling your own exception classes.
464
3176feef 465=item L<Error>
466
467Exception object implementation with a C<try> statement. Does not localize
468C<$@>.
469
470=item L<Exception::Class::TryCatch>
471
472Provides a C<catch> statement, but properly calling C<eval> is your
473responsibility.
474
475The C<try> keyword pushes C<$@> onto an error stack, avoiding some of the
1d64c1ad 476issues with C<$@>, but you still need to localize to prevent clobbering.
3176feef 477
478=back
479
faecd5a0 480=head1 LIGHTNING TALK
481
482I gave a lightning talk about this module, you can see the slides (Firefox
483only):
484
485L<http://nothingmuch.woobling.org/talks/takahashi.xul?data=try_tiny.txt>
486
487Or read the source:
488
489L<http://nothingmuch.woobling.org/talks/yapc_asia_2009/try_tiny.yml>
490
3176feef 491=head1 VERSION CONTROL
492
493L<http://github.com/nothingmuch/try-tiny/>
494
495=head1 AUTHOR
496
497Yuval Kogman E<lt>nothingmuch@woobling.orgE<gt>
498
499=head1 COPYRIGHT
500
c4e1eb12 501 Copyright (c) 2009 Yuval Kogman. All rights reserved.
3176feef 502 This program is free software; you can redistribute
c4e1eb12 503 it and/or modify it under the terms of the MIT license.
3176feef 504
505=cut
506