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