All expected evals converted to try, except where no test is done,
[dbsrgits/DBIx-Class.git] / lib / DBIx / Class / Storage / TxnScopeGuard.pm
1 package DBIx::Class::Storage::TxnScopeGuard;
2
3 use strict;
4 use warnings;
5 use Carp::Clan qw/^DBIx::Class/;
6 use Try::Tiny;
7
8 sub new {
9   my ($class, $storage) = @_;
10
11   $storage->txn_begin;
12   bless [ 0, $storage ], ref $class || $class;
13 }
14
15 sub commit {
16   my $self = shift;
17
18   $self->[1]->txn_commit;
19   $self->[0] = 1;
20 }
21
22 sub DESTROY {
23   my ($dismiss, $storage) = @{$_[0]};
24
25   return if $dismiss;
26
27   my $exception = $@;
28
29   {
30     local $@;
31
32     carp 'A DBIx::Class::Storage::TxnScopeGuard went out of scope without explicit commit or error. Rolling back.'
33       unless $exception;
34
35     my $rollback_exception;
36     try { $storage->txn_rollback }
37     catch { $rollback_exception = shift };
38
39     if (defined $rollback_exception && $rollback_exception !~ /DBIx::Class::Storage::NESTED_ROLLBACK_EXCEPTION/) {
40       if ($exception) {
41         $exception = "Transaction aborted: ${exception} "
42           ."Rollback failed: ${rollback_exception}";
43       }
44       else {
45         carp (join ' ',
46           "********************* ROLLBACK FAILED!!! ********************",
47           "\nA rollback operation failed after the guard went out of scope.",
48           'This is potentially a disastrous situation, check your data for',
49           "consistency: $rollback_exception"
50         );
51       }
52     }
53   }
54
55   $@ = $exception;
56 }
57
58 1;
59
60 __END__
61
62 =head1 NAME
63
64 DBIx::Class::Storage::TxnScopeGuard - Scope-based transaction handling
65
66 =head1 SYNOPSIS
67
68  sub foo {
69    my ($self, $schema) = @_;
70
71    my $guard = $schema->txn_scope_guard;
72
73    # Multiple database operations here
74
75    $guard->commit;
76  }
77
78 =head1 DESCRIPTION
79
80 An object that behaves much like L<Scope::Guard>, but hardcoded to do the
81 right thing with transactions in DBIx::Class. 
82
83 =head1 METHODS
84
85 =head2 new
86
87 Creating an instance of this class will start a new transaction (by
88 implicitly calling L<DBIx::Class::Storage/txn_begin>. Expects a
89 L<DBIx::Class::Storage> object as its only argument.
90
91 =head2 commit
92
93 Commit the transaction, and stop guarding the scope. If this method is not
94 called and this object goes out of scope (e.g. an exception is thrown) then
95 the transaction is rolled back, via L<DBIx::Class::Storage/txn_rollback>
96
97 =cut
98
99 =head1 SEE ALSO
100
101 L<DBIx::Class::Schema/txn_scope_guard>.
102
103 =head1 AUTHOR
104
105 Ash Berlin, 2008.
106
107 Inspired by L<Scope::Guard> by chocolateboy.
108
109 This module is free software. It may be used, redistributed and/or modified
110 under the same terms as Perl itself.
111
112 =cut