Tutorial DBIx::Class, conexao com banco de dados com perl ORM PT BR

Resumo

Neste artigo, dou várias dicas de como fazer determinadas coisas com dbix class. Este não vai ser fácil para iniciantes que não leram a documentação. 
No entanto, espero que sirva como referência.

Código Fonte

Faça o download do pacote e acesse o diretório 'app' para obter o código fonte do exemplo utilizado.

O que é o DBIx::Class ?

DBIx::Class é um extensível e flexível objeto <-> mapeamento relacional
Na minha opinião é o mais fantástico mapeamento objeto relacional. 

Com ele você consegue acessar tabelas relacionadas a partir de qualquer ponto. Ou seja, vamos supor que temos um db assim:

   ______             _______              _______                ________ 
  |      |           |       |            |       |              |        |
  |  pai |----------<| filho |-----------<| amigo |--------------|namorada|
  |______|  1 ou +   |_______|  1 ou +    |_______|  apenas 1    |________|

(pai tem 1 ou + filhos, cada filho pode ter 1 ou + amigos e cada amigo 
 pode ter 1 namorada )

Agora vamos supor que estamos usando dbix class e queremos adicionar:
- 1 pai, 
- 1 filho, 
- 2 amigos
- 2 namordas para cada amigo inserido

É bastante simples... veja como na seção de apêndice logo após as dicas de A a Z.

A. Como gerar os models se você já tem o banco de dados, com catalyst:

$ script/myapp_create.pl model DB DBIC::Schema MyApp::DB create=static dbi:Pg:dbname=myapp USER pass
$ script/myapp_create.pl model DB DBIC::Schema MyApp::DB create=static dbi:mysql:db=myapp USER pass

B. Como gerar os models se você já tem o banco de dados, sem catalyst:

dbicdump -o dump_directory=./lib \
-o components='["InflateColumn::DateTime"]' \   <-- *** nao obrigatorio
-o debug=1 \
My::Schema \
'dbi:Pg:dbname=foo' \
 myuser \
mypassword

C. Ex de script simplão (p/ executar via console) com dbix class (sem catalsyt):

1. fazer um schema dump com dbix class schema loader (letra B)
 2. colocar seu projeto em pastas assim:
/myapp
/myapp/programa.pl
/myapp/lib
/myapp/lib/DBSchema

3.editar seu programa.pl e adicionar estas linhas para ele poder utilizar o DBSchema:
 use lib ( "./lib" );
use DBSchema;  
my $schema = DBSchema->connect('dbi:Pg:dbname=saude', 'hernan', '123');
my $medico = $schema->resultset('Medico')->find({ id => 1});
print $medico->name;

D. Exemplo de cache direto no dbix class:

http://search.cpan.org/~rkitover/Catalyst-Model-DBIC-Schema-0.41/lib/Catalyst/TraitFor/Model/DBIC/Schema/Caching.pm

__PACKAGE__->config({
traits => ['Caching'],
connect_info => 
['dbi:mysql:db', 'user', 'pass'],
});

$c->model('DB::Table')->search({ foo => 'bar' }, { cache_for => 18000 });  

E. Exemplo de ResultSet (extendendo os models), nos permite jogar todas as lógicas nos models, gerando thin controllers fat models.

Depois é só acessar como se fosse um metodo.
infos: http://beta.metacpan.org/module/Catalyst::Model::DBIxResultSet

sub is_my_img {
my ($self, $c, $img_gallery_id) = @_;
 return $self->search({
id => $img_gallery_id,
user_id => $c->user->id,
});

No controller:
my $test = $c->model('DBICSchemamodel::ImgGallery')->is_my_img($c, $pks[0])->single();

F. Deploy de banco de dados, Tendo os models em mãos, é possível criar as tabelas (após conectar) num banco de dados:

$schema->deploy

G. Exemplo de coluna Count

dbic dbix class count 
search({}, {
order_by => { -desc => \'count(tracks.trackid)' },
join => 'tracks',
distinct => 1,
});  

H. Custom Query

my $schema = DB->connect(...);
my $stmt = 'create table foo ( id int );';
$schema->storage->dbh->prepare( $stmt )->execute()

I. Exemplo HashReinflator (hashreinflator devolve um hash ao inves de um resultset), com cache:

my $uniqueKey = md5_hex($schemamodel);
my $cached_data;
unless ( $cached_data = $c->cache->get($uniqueKey) ) {
my $result = $c->model($schemamodel)
 ->search({
  is_deleted => 0, 
 },{
 });
 $result->result_class('DBIx::Class::ResultClass::HashRefInflator'); #sets the result to be hashreinflator
 my @items_list = $result->all; #inflates the whole resultset into a array hash
  use Data::Dumper;
 $c->log->debug('Dumper ' . Dumper( \@items_list )); 
 $cached_data = \@items_list;
 $c->cache->set( $uniqueKey, $cached_data );
}

J. Insert multiplo:

$cd->artistname(shift @{$c->req->params->{artistname}});  
$cd->update(); 

K. Adicionando um metodo (no model/resultset) que retorna preço formatado

sub preco_fmt #retorna preço formatado
{
   my $self  = shift @_;
   return 'R$ ' . sub { my $price_fmt = $self->preco() ; $price_fmt =~ s/\./,/g; return $price_fmt; }->();
} 

L. DBIx::Class search NOT IN, -not_in, not in

->search({... 
id => { 
 'not in' => [1,2], 
 },
...},{}); 

M. Order by count

order_by => \''COUNT(\'story_id\') AS count'
order_by => \'COUNT(\'story_id\') DESC'

N. update user session data

$c->user->obj->discard_changes

O. Exemplo de um ResultSet Class para o model Company

package myapp::DBSchema::ResultSet::Company;
use strict;
use warnings;
use base 'DBIx::Class::ResultSet';

sub all_companys {
my ($self) = @_;
my @companys = $self->
 search( { 'me.is_deleted' => 0 }, { order_by => ['me.name ASC',] } )->all;
return @companys;
}

1;  

P. **** ATive o trace do dbix class para debugar as querys, etc...

sem cores:
   $ export DBIC_TRACE=1 && script/imobiliaria_software_server.pl -r -d -p 3089
com cores:
   $ export DBIC_TRACE_PROFILE=console && export DBIC_TRACE=1 && script/imobiliaria_software_server.pl -r -d -p 3089

Q. Query no dbix class:

$item = $schema->resultset('Medico')->find(); #retorna uma row, pode ser acessado ex. 
$item->id, 
$item->nome, 
$item->nome('novo nome') 
$item->update
@res = $schema->resultset('Medico')->search({},{})->all #Retorna array
$res = $schema->resultset('Medico')->search({...},{})  #Retorna varias rows e para fazer loop tem que fazer: while ( my $item = $res->next ) { ... } 

APÊNDICE

Veja como é bacana e elegante trabalhar com DBIx::Class

Crie seu banco de dados ( utilizei postgresql )

Eu recomento a utilização do postgres para ensinar dbix class pois o postgres salva os relacionamentos dentro do banco de dados e isso facilita pois o dbix class consegue detectar esses relacionamentos e já cria todos os models para nós incluindo esses relacionamentos.

CREATE TABLE amigo (
    id integer NOT NULL,
    nome text,
    amigo_id integer
);
ALTER TABLE public.amigo OWNER TO hernan;
CREATE SEQUENCE amigo_id_seq
    START WITH 1
    INCREMENT BY 1
    NO MINVALUE
    NO MAXVALUE
    CACHE 1;
ALTER TABLE public.amigo_id_seq OWNER TO hernan;
ALTER SEQUENCE amigo_id_seq OWNED BY amigo.id;
SELECT pg_catalog.setval('amigo_id_seq', 1, false);
CREATE TABLE filho (
    id integer NOT NULL,
    nome text,
    pai_id integer
);
ALTER TABLE public.filho OWNER TO hernan;
CREATE SEQUENCE filho_id_seq
    START WITH 1
    INCREMENT BY 1
    NO MINVALUE
    NO MAXVALUE
    CACHE 1;
ALTER TABLE public.filho_id_seq OWNER TO hernan;
ALTER SEQUENCE filho_id_seq OWNED BY filho.id;
SELECT pg_catalog.setval('filho_id_seq', 1, false);
CREATE TABLE namorada (
    id integer NOT NULL,
    nome text,
    amigo_id integer
);
ALTER TABLE public.namorada OWNER TO hernan;
CREATE SEQUENCE namorada_id_seq
    START WITH 1
    INCREMENT BY 1
    NO MINVALUE
    NO MAXVALUE
    CACHE 1;
ALTER TABLE public.namorada_id_seq OWNER TO hernan;
ALTER SEQUENCE namorada_id_seq OWNED BY namorada.id;
SELECT pg_catalog.setval('namorada_id_seq', 1, false);
CREATE TABLE pai (
    id integer NOT NULL,
    nome text
);
ALTER TABLE public.pai OWNER TO hernan;
CREATE SEQUENCE pai_id_seq
    START WITH 1
    INCREMENT BY 1
    NO MINVALUE
    NO MAXVALUE
    CACHE 1;
ALTER TABLE public.pai_id_seq OWNER TO hernan;
ALTER SEQUENCE pai_id_seq OWNED BY pai.id;
SELECT pg_catalog.setval('pai_id_seq', 1, false);
ALTER TABLE ONLY amigo ALTER COLUMN id SET DEFAULT nextval('amigo_id_seq'::regclass);
ALTER TABLE ONLY filho ALTER COLUMN id SET DEFAULT nextval('filho_id_seq'::regclass);
ALTER TABLE ONLY namorada ALTER COLUMN id SET DEFAULT nextval('namorada_id_seq'::regclass);
ALTER TABLE ONLY pai ALTER COLUMN id SET DEFAULT nextval('pai_id_seq'::regclass);
COPY amigo (id, nome, amigo_id) FROM stdin;
\.
COPY filho (id, nome, pai_id) FROM stdin;
\.
COPY namorada (id, nome, amigo_id) FROM stdin;
\.
COPY pai (id, nome) FROM stdin;
\.
ALTER TABLE ONLY amigo
    ADD CONSTRAINT amigo_pkey PRIMARY KEY (id);
ALTER TABLE ONLY filho
    ADD CONSTRAINT filho_pkey PRIMARY KEY (id);
ALTER TABLE ONLY namorada
    ADD CONSTRAINT namorada_pkey PRIMARY KEY (id);
ALTER TABLE ONLY pai
    ADD CONSTRAINT pai_pkey PRIMARY KEY (id);
ALTER TABLE ONLY amigo
    ADD CONSTRAINT amigo_amigo_id_fkey FOREIGN KEY (amigo_id) REFERENCES filho(id);
ALTER TABLE ONLY filho
    ADD CONSTRAINT filho_pai_id_fkey FOREIGN KEY (pai_id) REFERENCES pai(id);
ALTER TABLE ONLY namorada
    ADD CONSTRAINT namorada_amigo_id_fkey FOREIGN KEY (amigo_id) REFERENCES amigo(id);
REVOKE ALL ON SCHEMA public FROM PUBLIC;
REVOKE ALL ON SCHEMA public FROM postgres;
GRANT ALL ON SCHEMA public TO postgres;
GRANT ALL ON SCHEMA public TO nixus;
GRANT ALL ON SCHEMA public TO PUBLIC;

Como conectar no banco de dados com dbix class

Agora que já criamos o banco de dados, devemos criar os models em nossa app para que ela se conecte no banco de dados. Então vamos lá... usaremos a dica "B. Como gerar os models se você já tem o banco de dados, sem catalyst" que diz:

dbicdump -o dump_directory=./lib \
-o components='["InflateColumn::DateTime"]' \   <-- *** nao obrigatorio
-o debug=1 \
My::Schema \
'dbi:Pg:dbname=foo' \
 myuser \
mypassword

Se você não tiver permissão, pode conectar no db com

psql tut_dbixclass_perl_orm

e depois executar um grant all em todas as tabelas pra um usuario, no meu casi 'webdev'

select 'grant all on '||schemaname||'.'||tablename||' to webdev;' from pg_tables where schemaname in ('public') order by schemaname, tablename;

faça um select em pg_tables antes para ver quais itens você vai precisar... neste caso é só public e agora vamos criar um diretório para a aplicação

mkdir /home/catalyst/tutorial-dbix-class-perl-orm-pt-br/app
cd /home/catalyst/tutorial-dbix-class-perl-orm-pt-br/app
vim /home/catalyst/tutorial-dbix-class-perl-orm-pt-br/app/app.pl

Dependências

Instale o módulos do cpan:

DBIx::Class::Schema::Loader     #dbicdump (gera os models pra você, com todos os relacionamentos)
DBD::Pg                         #para conectar no banco de dados postgres

Talvez você precise daquelas libs de -dev para poder criar o DBD::Pg... ex.

postgresql-server-dev-all       #ferramentas para desenvolvimento postgres

ERRO#1 - verificando se o comando dbicdump está disponível

$ dbicdump
The program 'dbicdump' is currently not installed.  To run 'dbicdump' please ask your administrator to install the package 'libdbix-class-schema-loader-perl'

Primeiro tentei rodar o comando dbicdump diretamente no meu terminal para ver se o mesmo está instalado. Apareceu essa mensagem dizendo que o mesmo não está instalado, porem eu posso instalar pois ele está no repositório do ubuntu (que legal em) Mas, acontece que eu estou utilizando minha versão de perl 5.17.1 (a mais nova) que eu instalei usando "perlbrew" (procure perlbrew) e eu utilizo junto o "cpanm" e assim eu posso instalar módulos sem root no perl. legal pois é mais seguro!!

cpanm DBIx::Class::Schema::Loader

esse é o módulo que fornece o dbicdump

ERRO#2

Após instalar o DBIx::Class::Schema::Loader, Tentei executar o comando dbicdump mas veja o que aconteceu:

$ dbicdump -o dump_directory=./lib -o debug=1 DB::Tutorial::DBIx::Class::PT::BR 'dbi:Pg:dbname=tut_dbixclass_perl_orm' username password
DBIx::Class::Schema::Loader::make_schema_at(): DBI Connection failed: install_driver(Pg) failed: Can't locate DBD/Pg.pm in @INC (@INC contains: /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/site_perl/5.15.9/x86_64-linux /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/site_perl/5.15.9 /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/5.15.9/x86_64-linux /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/5.15.9 .) at (eval 95) line 3.
Perhaps the DBD::Pg perl module hasn't been fully installed,
or perhaps the capitalisation of 'Pg' isn't right.
Available drivers: DBM, ExampleP, File, Gofer, Proxy, SQLite, Sponge.
 at /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/site_perl/5.15.9/DBIx/Class/Storage/DBI.pm line 1249. at /home/webdev/perl5/perlbrew/perls/perl-5.15.9/bin/dbicdump line 178

Isto quer dizer que não tem instalado o DBD::Pg, que é o módulo que vai permitir nossa conexão com o banco de dados Postgres. O erro aparece na linha acima, nesta parte:

Can't locate DBD/Pg.pm in @INC

Quer dizer que não localizou o módulo DBD/Pg e esse módulo é uma dependência necessária para essa ação:

DBIx::Class::Schema::Loader::make_schema_at(): DBI Connection failed

Ou seja, tentou executar metodo:

make_schema 

no módulo

DBIx::Class::Schema::Loader

e resultou em:

DBI Connection failed 

pelo motivo

install_driver(Pg) failed: Can't locate DBD/Pg.pm in @INC

Arrumando ERRO#2 (Instalando DBD::Pg)

Quando tentei rodar o dbicdump, eu havia instalado minha versão perl com perlbrew recentemente e nem tinha instalado o DBD::Pg ainda... por isso resultou no erro #2. Então vou instalar o módulo DBD::Pg assim:

$ cpanm DBD::Pg
--> Working on DBD::Pg
Fetching http://www.cpan.org/authors/id/T/TU/TURNSTEP/DBD-Pg-2.19.2.tar.gz ... OK
Configuring DBD-Pg-2.19.2 ... OK
Building and testing DBD-Pg-2.19.2 ... OK
Successfully installed DBD-Pg-2.19.2
1 distribution installed

ERRO#4 Usuário sem permissão no banco de dados

$ dbicdump -o dump_directory=./lib -o debug=1 DB::Tutorial::DBIx::Class::PT::BR 'dbi:Pg:dbname=tut_dbixclass_perl_orm' username password
DBIx::Class::Schema::Loader::make_schema_at(): DBI Connection failed: DBI connect('dbname=tut_dbixclass_perl_orm','username',...) failed: FATAL:  Peer authentication failed for user "username" at /home/webdev/perl5/perlbrew/perls/perl-5.15.9/lib/site_perl/5.15.9/DBIx/Class/Storage/DBI.pm line 1249. at /home/webdev/perl5/perlbrew/perls/perl-5.15.9/bin/dbicdump line 178

Dê permissão ao seu usuário no banco de dados atraves do comando grant

grant all on database tut_dbixclass_perl_orm to username;

ERRO#5 Usuário sem permissão de select

$ dbicdump -o dump_directory=./lib -o debug=1 DB::Tutorial::DBIx::Class::PT::BR 'dbi:Pg:dbname=tut_dbixclass_perl_orm' username webdev123
Bad table or view 'amigo', ignoring: DBIx::Class::Schema::Loader::make_schema_at(): DBI Exception: DBD::Pg::st execute failed: ERROR:  permission denied for relation amigo [for Statement "SELECT * FROM "public"."amigo" WHERE ( 1 = 0 )"] at /home/webdev/perl5/perlbrew/perls/perl-5.15.9/bin/dbicdump line 178

dê permissão de select nas tabelas para seu usuário:

grant all on public.pai to username;

Quando o dbicdump dá certo voce vê:

$ dbicdump -o dump_directory=./lib -o debug=1 DB::Tutorial::DBIx::Class::PT::BR 'dbi:Pg:dbname=tut_dbixclass_perl_orm' username password
DB::Tutorial::DBIx::Class::PT::BR::Result::Amigo->table("amigo");
DB::Tutorial::DBIx::Class::PT::BR::Result::Amigo->add_columns(
  "id",
  {
    data_type         => "integer",
    is_auto_increment => 1,
    is_nullable       => 0,
    sequence          => "amigo_id_seq",
  },
  "nome",
  { data_type => "text", is_nullable => 1 },
  "amigo_id",
  { data_type => "integer", is_foreign_key => 1, is_nullable => 1 },
);
DB::Tutorial::DBIx::Class::PT::BR::Result::Amigo->set_primary_key("id");
DB::Tutorial::DBIx::Class::PT::BR::Result::Filho->table("filho");
DB::Tutorial::DBIx::Class::PT::BR::Result::Filho->add_columns(
  "id",
  {
    data_type         => "integer",
    is_auto_increment => 1,
    is_nullable       => 0,
    sequence          => "filho_id_seq",
  },
  "nome",
  { data_type => "text", is_nullable => 1 },
  "pai_id",
  { data_type => "integer", is_foreign_key => 1, is_nullable => 1 },
);
DB::Tutorial::DBIx::Class::PT::BR::Result::Filho->set_primary_key("id");
DB::Tutorial::DBIx::Class::PT::BR::Result::Namorada->table("namorada");
DB::Tutorial::DBIx::Class::PT::BR::Result::Namorada->add_columns(
  "id",
  {
    data_type         => "integer",
    is_auto_increment => 1,
    is_nullable       => 0,
    sequence          => "namorada_id_seq",
  },
  "nome",
  { data_type => "text", is_nullable => 1 },
  "amigo_id",
  { data_type => "integer", is_foreign_key => 1, is_nullable => 1 },
);
DB::Tutorial::DBIx::Class::PT::BR::Result::Namorada->set_primary_key("id");
DB::Tutorial::DBIx::Class::PT::BR::Result::Pai->table("pai");
DB::Tutorial::DBIx::Class::PT::BR::Result::Pai->add_columns(
  "id",
  {
    data_type         => "integer",
    is_auto_increment => 1,
    is_nullable       => 0,
    sequence          => "pai_id_seq",
  },
  "nome",
  { data_type => "text", is_nullable => 1 },

... e nada de erros.

Agora precisamos arrumar alguns relacionamentos

O DBIx::Class fala ingles por padrao.. então quando ele detectar e criar os relacionamentos para você, ele vai colocar os plurais em ingles. Então precisamos verificar e consertar essas inconsistências.

Então vamos editar os models do dbixclass que estão dentro do diretório:

app/lib/DB/Tutorial/DBIx/Class/PT/BR/Result/*
app/lib/DB/Tutorial/DBIx/Class/PT/BR/Result/Amigo.pm
app/lib/DB/Tutorial/DBIx/Class/PT/BR/Result/Filho.pm
app/lib/DB/Tutorial/DBIx/Class/PT/BR/Result/Pai.pm

Primeiro edite o Filho.pm la no final, *após* a linha DO NOT MODIFY THIS OR ANYTHING ABOVE coloque: Não altere essa linha e nem o conteudo acima dela. Tudo que você escrever após essa linha vai sobrepor o que foi declarado em cima. A parte que está em cima pode ser sobre-escrita/atualizada se você rodar um comando para atualizar os campos do banco de dados por exemplo. Então se você mexer ali, o dbix class vai se perder, então evite mexer ali para não ter problemas. Sempre mexa após essa linha:

# DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:aaQcKvggMUd5YmFttR/eYw

__PACKAGE__->has_many(
  "amigos",
  "DB::Tutorial::DBIx::Class::PT::BR::Result::Amigo",
  { "foreign.amigo_id" => "self.id" },
  { cascade_copy => 0, cascade_delete => 0 },
);

agora o arquivo Pai.pm

# DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:B4I5N4/abdMhSMNWiYJFJQ

__PACKAGE__->has_many(
  "filhos",
  "DB::Tutorial::DBIx::Class::PT::BR::Result::Filho",
  { "foreign.pai_id" => "self.id" },
  { cascade_copy => 0, cascade_delete => 0 },
);

agora sim, alteramos de filhoes para filhos. E amigoes para amigos. bem melhor pois fica mais natural.

Como criar a aplicação com dbix class e conexão com banco de dados postgres

Veja a dica #C deste tutorial e crie um diretório para sua aplicação e edite um arquivo.pl para inserir o seguinte código de exemplo: Este código conecta no banco de dados utilizando postgres, e insere 1 pai, 1 filho, 1 amigo e uma namorada pro amigo. Abaixo está a saída dos comandos executados. Para executar o script digite no terminal:

$ export DBIC_TRACE=1 && perl app.pl

Segue o codigo fonte:

use lib ( "./lib" );
use DB::Tutorial::DBIx::Class::PT::BR; 
my $schema = DB::Tutorial::DBIx::Class::PT::BR->connect(
    'dbi:Pg:dbname=tut_dbixclass_perl_orm', 
    'webdev', 
    'webdev123'
);
my $pai = $schema->resultset('Pai')->new({ nome => 'joao' }); 
$pai->insert;
warn $pai->nome;
my $filho = $pai->add_to_filhos( { nome => 'filho 1' } );
warn $filho->nome;
my $amigo = $filho->add_to_amigos( {
    nome => 'Nome amigo1',
} );

my $namorada = $amigo->add_to_namoradas( {
    nome => 'Maria' 
} );

warn $namorada->nome;
warn $namorada->id;

Saida do codigo acima:

INSERT INTO pai ( nome) VALUES ( ? ) RETURNING id: 'joao'
joao at app.pl line 10.
INSERT INTO filho ( nome, pai_id) VALUES ( ?, ? ) RETURNING id: 'filho 1', '23'
filho 1 at app.pl line 12.
INSERT INTO amigo ( amigo_id, nome) VALUES ( ?, ? ) RETURNING id: '6', 'Nome amigo1'
INSERT INTO namorada ( amigo_id, nome) VALUES ( ?, ? ) RETURNING id: '3', 'Maria'
Maria at app.pl line 21.
2 at app.pl line 22.

Autor

Hernan Lopes < hernanlopes gmail >
cpan: http://search.cpan.org/~hernan/
github:  http://github.com/hernan604/

Escrevam para eu saber se está ficando legal, ou como posso melhorar! 
Obg, 
-Hernan Lopes

1; # The preceding line will help the module return a true value