NAME
Class::Entity - Object interface for relational databases
DESCRIPTION
Class::Entity allows a developer to create an object interface for a relational database by writing a minimal amount of code in a set of sub-classes which correspond to database tables.
Right now this module only implements a read only interface. Writes will probably come later.
SYNOPSIS
package Table;
use base qw(Class::Entity);
package main;
use DBI;
my $dbh = DBI->connect(...);
my $table = Table->fetch(dbh => $dbh, value => 1234);
print $table->Column;
for (Table->find(dbh => $dbh, value => "Name like 'foo%'")) {
printf "% %\n", $table->Name, $table->Date;
}
METHODS
- new(dbh => $dbh, data => $data)
-
The constructor creates a new instance of the current class. If you pass in values via data they will be stored in the object for later use.
- _primary_key()
-
Return the primary key for the table the current sub-class of Class::Entity represents. You will normally want to overide this in the sub-class but for convenience it returns id by default.
The value of this method is used to create the query that is run when a call to fetch is made.
- _table()
-
Return the name of the table the current sub-class of Class::Entity represents. You will normally want to overide this in the sub-class but for convenience it returns the final part of the sub-class name; that is: the sub-class name minus the leading /.*:/.
- _relation()
-
This method provides the sub-class author a means of joing accross multiple tables when a call to fetch or find is made. All database fields returned via these methods are stored in an instance of the current sub-class and exposed via the autoloader. By default it returns all fields in the current table represented by the sub-class.
- _object_map()
-
The object map works in conjuction with the autoloader. If you have sub-classes of Class::Entity, which represent tables linked to in the current sub-class, you can overload this method to return a hash where the keys are the table columns and the values are the names of the associated Class::Entity sub-classes.
The object map will only be used for method calls of the form get_Value.
Here's an example:
package Users; use base qw(Class::Entity); package Departments; use base qw(Class::Entity); sub _object_map({ UserID => "Users" }) package main; use DBI; my $dbh DBI->connect(...); my @support = Departments->find(dbh => $dbh, where => "Name = 'Support'"); for (@support) { printf "%s %s\n", $_->UserID, $_->get_UserID->name; }
- fetch(dbh => $dbh, key => $key)
-
Return an instance of the current sub-class. You must provide a database value via dbh and a primary key value via key. The database handle is stored in the returned object for later use and all table fields are exposed via the auoloader.
- find(dbh => $dbh, where => $where)
-
Return an array in array context, or the first item of the array in scalar context, of instances of the current sub-class based on the query modifier passed in via where. You must pass in a database handle via dbh which will be stored in the returned instances for later use.
- AUTOLOAD([$value])
-
The autoloader provides get and set methods for the table values represented in an instance of the current sub-class. For example, if you have a table with the fields: Name, Date, Subject, you would access them like this:
package Table; use base qw(Class::Entity); package main; use DBI; my $dbh = DBI->connect(...); my $table = Table->fetch(dbh => $dbh, key => 10); print $table->Name . "\n"; print $table->Date . "\n"; print $table->Subject . "\n";
If you call an anonymous method of the form get_Value, where Value is a column represented by the current object, the autoloader will attempt to dispatch the call to the fetch method of the corresponding sub-class of Class::Entity, if it's listed in the _bject_map.
SEE ALSO
DBI
AUTHORS
Paddy Newman, <pnewman@cpan.com>
ACKNOWLEDGEMENTS
This is basically a cut-down, slightly modified version of something an ex-colegue of mine wrote and introduced me to. His name is Dan Barlow and he's a much better programmer than me and he deserves all the credit.