NAME
DBIx::Struct - convenience SQL functions with Class::Struct-like row objects
SYNOPSIS
use DBIx::Struct;
DBIx::Struct::connect($data_source, $username, $auth);
my $row = one_row("table", $idField);
print $row->field;
$row->field('new data');
$row->update;
my $rows = all_rows("table", {field => "some data"});
print $rows->[0]->field;
DESCRIPTION
Makes SQL queries from Perl data structures. It uses SQL::Abstract module to parse "where" and "order by" structures. It also supports "group by", "having", "limit", "offset" and mapping function. To make actual queries subclass of DBIx::Connector is used. The main purpose of this module is to provide a very easy, simple and efficient interface to database. Internally every row is just an array of values and every accessor knows its offset in that array. This is much more efficient as storing row in hash.
IMPORT
This module can be integrated into some framework that already uses DBIx::Connector. Integration parameters should be passed on first use. These parameters are used for integration:
- connector_module
-
Connector's module name. By default used its own connector.
- connector_constructor
-
Connector's constructor function name. By default is 'new'.
- connector_args
-
Connector's arguments: ($dsn, $user, $password, $connect_attrs)
- connector_object
-
Connector's singleton object. By defaut used its own object.
- table_classes_namespace
-
Namespace for table classes. By default is 'DBC'.
- query_classes_namespace
-
Namespace for query classes. By default is 'DBQ'.
- connect_timeout
-
Timeout to connect or reconnect to DB for default connector.
- error_class
-
Errors can be reported as strings or as hashes. By default errors reported as strings. To report error as string 'DBIx::Struct::Error::String' is used. There's 'DBIx::Struct::Error::Hash' to receive a structured hash containg keys: result => 'SQLERR', message => $message and probably some others.
This module always exports three functions: one_row()
, all_rows()
and new_row()
. Optionally it can export connector()
- connector's singleton object and hash_ref_slice()
function.
Possible usage:
use DBIx::Struct (
connector_module => 'PEF::Front::Connector',
error_class => 'DBIx::Struct::Error::Hash'
);
use DBIx::Struct qw(connector hash_ref_slice);
use DBIx::Struct;
METHODS
hash_ref_slice($hashref, @keys)
Simple hash slice function before perl-5.20 is widely distributred. It accepts hash reference and array of keys. Exportable. Returns array of key-value pairs.
DBC::Article->delete({hash_ref_slice $req, 'id_article'});
is the same as
DBC::Article->delete({id_article => $req->{id_article}});
And
new_row(
article => hash_ref_slice $req,
qw(title content id_author)
);
is the same as
new_row('article',
title => $req->{title},
content => $req->{content},
id_author => $req->{id_author}
);
connect($data_source, $username, $auth)
This method connects to database or returns already connected connector object. For every table in database there's one corresponding class in DBC:: namespace in CamelCase style after successful connect created. E.g. DBC::ClientData is created for table "client_data". This namespace is configurable.
one_row($table, $abstract_where, $order_by, $limit, $offset)
Selects one row from given table and returns its object.
$table
can be a simple table name, SQL query beginning with "select" and ending with SQL-from clause or array reference to generate SQL from Perl data structures.
For array reference following arguments can be specified:
$table_name
Any scalar without leading '-' means table name
-columns => \@columns
or
-columns => $column
\@columns is a reference to an array of SQL column specifications (i.e. column names, * or table.*, functions, etc.). '*' is by default.
-left => $table
left [OUTER] join
-right => $table
right [OUTER] join
-join => $table
[INNER] join
-on => $join_condition_string
on($join_condition_string)
-using => $join_condition_string
using($join_field_string)
my $count = one_row([table => "count(*)"],
{ date => {'>=', \"now() - interval '1 hour'"})
->count;
This module is smart enough to determine automatically table connections.
my $name = one_row([client => session => -columns => "name"],
{ session_key => $input->{session}})
->name;
There are some named paramters can be used together with positional paramaters:
- -group_by
-
Generates "GROUP BY" SQL-clause
my $row1 = one_row("table", -group_by => "field");
Generates and executes "select * from table group by field" SQL-statement.
my $row2 = one_row("table", -group_by => [qw|field field2|]);
Generates and executes "select * from table group by field, field2" SQL-statement. Returns one row object.
- -having
-
Adds "HAVING" conditions to "GROUP BY"
my $row1 = one_row("table", -group_by => "field", -having => {"length(field)" => {">", 5} });
- -order_by
-
Generates "ORDER BY" SQL-clause as in SQL::Abstract.
- -where
-
Generates "WHERE" SQL-clause as in SQL::Abstract. If $abstract_where is just a scalar then it is supposed to be a primary key value.
- -limit
- -offset
-
Generates "LIMIT $X OFFSET $Y" SQL-clause.
all_rows($table, $abstract_where, $order_by, $limit, $offset)
This is just like previous one_row
but returns array of row objects. It has one more "floating" parameter: you can specify CODE reference in any place and it will be used as "mapping function".
my $rows = all_rows("table",
{ date => {'>=', \"now() - interval '1 hour'"},
sub {$_->filter_timestamp; $_->data});
This anonymous function is called for every row clearing timestamps from microseconds part and converting every row object into anonymous hash.
It's possible to use all_rows()
and one_row()
not only for simple table select but for queries also.
my $count = one_row("select count(*) from table",
{ date => {'>=', \"now() - interval '1 hour'"})
->count;
my $name = one_row("select name from client"
." join session on (id = id_client)",
{ session_key => $input->{session}})
->name;
new_row("table", column => $value, column2 => $value2, ...)
Inserts new row into table. Returns row object with set primary columns and passed in new_row
subroutine. If you need to set all columns from table then:
my $row = new_row("table", column => $value);
$row->fetch;
Row object methods
new
Creates new row in the table. This is the same as new_row()
except table name is already known from class name.
my $row = DBC::Table->new(email => 'a@bb.com', password => '12345');
set
Sets values of table columns. Returns object's $self value. Has three forms:
set([...])
-
Sets internal row data.
set({column => $value, ... })
-
Sets column(s) value(s)
set(column => $value, ... )
-
Same as above but without anonymous hash.
This method is useful when you can't use accessors. For example, when you have column name 'new' in your table. Full list of reserved keywords:
- new
- set
- data
- delete
- fetch
- update
- filter_timestamp
data
Returns values of the row. Has four forms:
data()
-
No parameters means to return all columns in anonymous hash {column => "value", ... }.
data([])
-
Empty array reference means internal row data.
data([qw|column column2|])
-
Non-empty array means array of values for corresponding columns.
data({column => undef, columnt2 => undef})
-
Returns given columns set in anonymous hash {column => "value", ... }.
update
Updates its row or table
update()
-
Updates changed (if any) columns of the row. Returns object's $self value.
update({column => "value"}, $where)
-
Updates table. For example:
DBC::List->update({ref => 33}, {id => 1});
Means
update list ref = ? WHERE ( id = ? )
with bind values 33, 1
delete
Deletes the row or rows from the table.
delete()
-
Deletes the row if the row has primary key.
delete($where)
-
deletes row(s) from table. For example:
DBC::List->delete({id => 1});
Means
delete from list WHERE ( id = ? )
with bind value 1
filter_timestamp
Removes microseconds from timestamp columns. Returns object's $self value.
2010-05-19 23:30:01.737126 -> 2010-05-19 23:30:01
fetch
Fetches the whole row data from table. Useful when primary key is known after insert but other columns are set by database. Returns object's $self value.
Accessors
All column names from table except special keywords are accessible via accessors:
my $prim = one_row("prim", 1);
print "payload: " . $prim->payload . "\n";
# new data
$prim->payload("pay never");
$prim->update;
Here $prim->payload("pay never") sets new column value. To store it explicitly call $prim->update. By default update is called on row object destruction. To include a random snippet of SQL verbatim, you specify it as a scalar reference.
$client->bonus(\"bonus + 10");
Or with parameters:
$client->bonus([\"bonus + ?", 10]);
This literal SQL expressions can be used for insert values also.
Syntax sugar
Referenced tables
Suppose you have two tables employer and employee:
employer:
id_employer,
name
employee:
id_employee,
id_employer references employer (id_employer),
id_employee_invited_by
name
alter table employee add constraint fk_employee_employee
foreign key (id_employee_invited_by) references employee (id_employee);
my $employee = one_row("employee", {name => 'John'});
my $employer = $employee->Employer;
Actually, $employee->Employer
is just the same as one_row("employer", $employee->id_employer)
.
Referenced by table
Now you want to find all people invited by one employee:
my $referenced_by = $employee->refEmployeeInvitedBys;
Or only those with name 'Robert':
my $robert_associates = $employee->Employer->refEmployees(name => 'Robert');
These names "refEmployees", "refEmployee" and alike are made up from:
"ref" . $TableName . $suffix . $plural
$plural
is "s" for plural (all_rows
) and empty for singular (one_row
). $suffix
is used when foreign key column name contains something more then just referenced table name and "_id" or "id_".
id_employer
of employee
makes methods refEmployee
and refEmployees
in class DBC::Employer
. And id_employee_invited_by
makes methods refEmployeeInvitedBy
and refEmployeeInvitedBys
.
Real-world usage scenarios
sub get_articles {
my $req = $_[0];
my $articles = all_rows(
[ # tables are automatically joined by FK->PK
"article a" => -join => "author w",
-columns => ['a.*', 'w.name author']
],
-order_by => {-desc => 'id_article'},
-limit => $req->{limit},
-offset => $req->{offset},
# strip miliseconds from timestamps and return anonymous hash for every row
sub { $_->filter_timestamp->data }
);
for my $article (@$articles) {
$article->{comment_count} =
# second unnamed argument is "where"
one_row([comment => -columns => 'count(*)'], {id_article => $article->{id_article}})->count;
}
return {
result => "OK",
articles => $articles,
count => one_row([article => -columns => 'count(*)'])->count
};
}
sub get_article_with_comments {
my $req = $_[0];
# second unnamed argument is "where" and plain scalar means it is primary key value
my $article = one_row(article => $req->{id_article});
return {
result => "NO_ARTICLE",
answer => "No such article"
} unless $article;
# transform object into hash
my $article_hash = $article->filter_timestamp->data;
# get author's name by executing one_row("author", { id_author => $article->id_author })->name
$article_hash->{author} = $article->Author->name;
return {
result => "OK",
article => $article_hash,
# execute complex recursive query
comments => connector->run(
sub {
$_->selectall_arrayref(
q{
with recursive article_comments(depth, path) as (
select 1 depth, array[id_comment] path, id_comment,
id_comment_parent, comment, author,
date_trunc('seconds', pub_date) pub_date
from comment
where id_article = ? and id_comment_parent is null
union all
select depth + 1, path || array[c.id_comment] path, c.id_comment,
c.id_comment_parent, c.comment, c.author,
date_trunc('seconds', c.pub_date) pub_date
from comment c, article_comments cs
where c.id_comment_parent = cs.id_comment
) select * from article_comments order by path, id_comment
},
{Slice => {}},
$req->{id_article}
);
}
)
};
}
Known issues
This module is tested only with PostgreSQL database. If you need other database support be prepared to help me to test and debug it.
See also
Authors
This module was written and is maintained by:
PEF Developer <pef-secure@yandex.ru>