The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
use t::Utils;
use xt::Utils::mysql;
use Test::More;
use Mock::Basic;

my $dbh = t::Utils->setup_dbh;
Mock::Basic->prepare_db($dbh);

subtest 'do basic transaction' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    $db->txn_begin;
    my $row = $db->insert('mock_basic',{
        name => 'perl',
    });
    is $row->id, 1;
    is $row->name, 'perl';
    $db->txn_commit;
    
    is $db->single('mock_basic',{id => 1})->name, 'perl';
};
 
subtest 'do rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    $db->txn_begin;
    my $row = $db->insert('mock_basic',{
        name => 'perl',
    });
    is $row->id, 2;
    is $row->name, 'perl';
    $db->txn_rollback;
    
    ok not +$db->single('mock_basic',{id => 2});
};

subtest 'do commit' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    $db->txn_begin;
    my $row = $db->insert('mock_basic',{
        name => 'perl',
    });
    is $row->id, 3;
    is $row->name, 'perl';
    $db->txn_commit;
 
    ok +$db->single('mock_basic',{id => 3});
};
 
subtest 'do scope commit' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    my $row = $db->insert('mock_basic',{
        name => 'perl',
    });
    is $row->id, 4;
    is $row->name, 'perl';
    $txn->commit;
 
    ok +$db->single('mock_basic',{id => 4});
    
};
 
subtest 'do scope rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    my $row = $db->insert('mock_basic',{
        name => 'perl',
    });
    is $row->id, 5;
    is $row->name, 'perl';
    $txn->rollback;
 
    ok not +$db->single('mock_basic',{id => 5});
    
};
 
subtest 'do scope guard for rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    {
        local $SIG{__WARN__} = sub {};
        my $txn = $db->txn_scope;
        my $row = $db->insert('mock_basic',{
            name => 'perl',
        });
        is $row->id, 6;
        is $row->name, 'perl';
    }
 
    ok not +$db->single('mock_basic',{id => 6});
    
};

subtest 'do nested scope rollback-rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 7;
        is $row2->name, 'perl5.10';
        $txn2->rollback;
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 8;
    is $row->name, 'perl5.12';
    $txn->rollback;

    ok not +$db->single('mock_basic',{id => 7});
    ok not +$db->single('mock_basic',{id => 8});
    
};

subtest 'do nested scope commit-rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 9;
        is $row2->name, 'perl5.10';
        $txn2->commit;
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 10;
    is $row->name, 'perl5.12';
    $txn->rollback;

    ok not +$db->single('mock_basic',{id => 9});
    ok not +$db->single('mock_basic',{id => 10});
    
};

subtest 'do nested scope rollback-commit' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 11;
        is $row2->name, 'perl5.10';
        $txn2->rollback;
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 12;
    is $row->name, 'perl5.12';

    eval { $txn->commit };

    like( $@, qr/tried to commit but already rollbacked in nested transaction./, "error message" );

    $txn->rollback;

    ok not +$db->single('mock_basic',{id => 11});
    ok not +$db->single('mock_basic',{id => 12});
    
};

subtest 'do nested scope commit-commit' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 13;
        is $row2->name, 'perl5.10';
        $txn2->commit;
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 14;
    is $row->name, 'perl5.12';
    $txn->commit;

    ok +$db->single('mock_basic',{id => 13});
    ok +$db->single('mock_basic',{id => 14});
    
};

subtest 'do nested scope rollback-commit-rollback' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        local $SIG{__WARN__} = sub {};
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 15;
        is $row2->name, 'perl5.10';

        {
            my $txn3 = $db->txn_scope;
            my $row3 = $db->insert('mock_basic',{
                name => 'perl',
            });
            is $row3->id, 16;
            is $row3->name, 'perl';
        }

        eval { $txn2->commit };
        like( $@, qr/tried to commit but already rollbacked in nested transaction./, "error message" );
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 17;
    is $row->name, 'perl5.12';
    $txn->rollback;

    ok not +$db->single('mock_basic',{id => 15});
    ok not +$db->single('mock_basic',{id => 16});
    ok not +$db->single('mock_basic',{id => 17});
    
};

subtest 'do nested scope rollback-commit-commit' => sub {
    my $db = Mock::Basic->new(dbh => $dbh);
    my $txn = $db->txn_scope;
    {
        local $SIG{__WARN__} = sub {};
        my $txn2 = $db->txn_scope;
        my $row2 = $db->insert('mock_basic',{
            name => 'perl5.10',
        });
        is $row2->id, 18;
        is $row2->name, 'perl5.10';

        {
            my $txn3 = $db->txn_scope;
            my $row3 = $db->insert('mock_basic',{
                name => 'perl',
            });
            is $row3->id, 19;
            is $row3->name, 'perl';
        }
    }
    my $row = $db->insert('mock_basic',{
        name => 'perl5.12',
    });
    is $row->id, 20;
    is $row->name, 'perl5.12';

    eval { $txn->commit };
    like( $@, qr/tried to commit but already rollbacked in nested transaction./, "error message" );
    $txn->rollback;

    ok not +$db->single('mock_basic',{id => 18});
    ok not +$db->single('mock_basic',{id => 19});
    ok not +$db->single('mock_basic',{id => 20});
    
};

done_testing;