DBIx-Poggy
view release on metacpan or search on metacpan
lib/DBIx/Poggy.pm view on Meta::CPAN
use strict;
use warnings;
use v5.14;
package DBIx::Poggy;
our $VERSION = '0.08';
use Scalar::Util qw(weaken refaddr);
=head1 NAME
DBIx::Poggy - async Pg with AnyEvent and Promises
=head1 SYNOPSIS
use strict;
use warnings;
use DBIx::Poggy;
my $pool = DBIx::Poggy->new( pool_size => 5 );
$pool->connect('dbi:Pg:db=test', 'root', 'password');
use AnyEvent;
my $cv = AnyEvent->condvar;
my $res;
$pool->take->selectrow_arrayref(
'SELECT * FROM users WHERE name = ?', {}, 'ruz'
)
->then(sub {
my $user = $res->{user} = shift;
return $pool->take->selectall_arrayref(
'SELECT * FROM friends WHERE user_id = ?', undef, $user->{id}
);
})
->then(sub {
my $friends = $res->{friends} = shift;
...
})
->catch(sub {
my $error = shift;
die $error;
})
->finally(sub {
$cv->send( $res );
});
$cv->recv;
=head1 DESCRIPTION
"Async" postgres as much as L<DBD::Pg> allows with L<Promises> instead of callbacks.
You get DBI interface you used to that returns promises, connections pool, queries
queuing and support of transactions.
=head2 Why pool?
DBD::Pg is not async, it's non blocking. Every connection can execute only one query
at a moment, so to execute several queries in parallel you need several connections.
What you get is you can do something in Perl side while postgres crunches data for
you.
=head2 Queue
Usually if you attempt to run two queries on the same connection then DBI throws an
error about active query. Poggy takes care of that by queuing up queries you run on
one connection. Handy for transactions and pool doesn't grow too much.
=head2 What is async here then?
Only a queries on multiple connections, so if you need to execute many parallel
queries then you need many connections. pg_bouncer and haproxy are your friends.
=head2 Pool management
In auto mode (default) you just "loose" reference to database handle and it gets
released back into the pool after all queries are done:
{
my $cv = AnyEvent->condvar;
$pool->take->do(...)->finally($cv);
$cv->recv;
}
# released
Or:
{
my $cv = AnyEvent->condvar;
my $dbh = $pool->take;
$dbh->do(...)
->then(sub { $dbh->do(...) })
->then(sub { ... })
->finally($cv);
$cv->recv;
}
# $dbh goes out of scope and all queries are done (cuz of condvar)
# released
=cut
use DBIx::Poggy::DBI;
use DBIx::Poggy::Error;
=head1 METHODS
=head2 new
Named arguments:
=over 4
( run in 0.697 second using v1.01-cache-2.11-cpan-140bd7fdf52 )