NAME
Coat::Persistent -- Simple Object-Relational mapping for Coat objects
DESCRIPTION
Coat::Persistent is an object to relational-databases mapper, it allows you to build instances of Coat objects and save them into a database transparently.
You basically define a mapping rule, either global or per-class and play with your Coat objects without bothering with SQL for simple cases (selecting, inserting, updating).
Coat::Peristent lets you use SQL if you want to, considering SQL is the best language when dealing with compelx queries.
WHY THIS MODULE ?
There are already very good ORMs for Perl available in the CPAN so why did this module get added?
Basically for one reason: I wanted a very simple way to build persistent objects for Coat and wanted something near the smart design of Rails'ORM (ActiveRecord). Moreover I wanted my ORM to let me send SQL requests if I wanted to (so I can do basic actions without SQL and complex queries with SQL).
This module is the result of my experiments of mixing DBI and Coat together, although it is a developer release, it works pretty well and fit my needs.
This module is expected to change in the future (don't consider the API to be stable at this time), and to grow (hopefully).
The underlying target of this module is to port the whole ActiveRecord::Base API to Perl. If you find the challenge and the idea interesting, feel free to contact me for giving a hand.
This is still a development version and should not be used in production environment.
DATA BACKEND
The concept behing this module is the same behind the ORM of Rails : all your tables must have a primary key named id. This may become configurable in future versions, but in this developer release this is not.
Your table names must be named like the package they map, with the following rules applied : lower case, replace "::" by "_". For instance a class Foo::Bar should be mapped to a table named "foo_bar".
All foreign key must be named "<table>_id" where table is the name if the class mapped formated like said above.
CONFIGURATION
DBI MAPPING
You have to tell Coat::Persistent how to map a class to a DBI driver. You can either choose to define a default mapper (in most of the cases this is what you want) or define a mapper for a specific class.
- Coat::Persistent->map_to_dbi $driver, @options
-
This will set the default mapper. Every class that hasn't a specific mapper set will use this one.
- __PACKAGE__->map_to_dbi $driver, @options
-
This will set a mapper for the current class.
Supported values for $driver are the following :
- csv : this will use DBI's "DBD:CSV" driver to map your instances to a CSV file. @options must contains a string as its first element being like the following: "f_dir=<DIRECTORY>" where DIRECTORY is the directory where to store de CSV files.
-
Example:
packahe Foo; use Coat::Persistent; __PACKAGE__->map_to_dbi('csv', 'f_dir=./t/csv-directory');
- mysql : this will use DBI's "dbi:mysql" driver to map your instances to a MySQL database. @options must be a list that contains repectively: the database name, the database user, the database password.
-
Example:
package Foo; use Coat::Persistent; __PACKAGE__->map_to_dbi('mysql' => 'dbname', 'dbuser', 'dbpass' );
CACHING
Since version 0.0_0.2, Coat::Persistent provides a simple way to cache the results of underlying SQL requests. By default, no cache is performed.
You can either choose to enable the caching system for all the classes (global cache) or for a specific class. You could also define different cache configurations for each class.
When the cache is enabled, every SQL query generated by Coat::Persistent is first looked through the cache collection. If the query is found, its cached result is returned; if not, the query is executed with the appropriate DBI mapper and the result is cached.
The backend used by Coat::Persistent for caching is Cache::FastMmap which is able to expire the data on his own. Coat::Persistent lets you access the Cache::FastMmap object through a static accessor :
- Coat::Persistent->cache : return the default cache object
- __PACKAGE__->cache : return the cache object for the class __PACKAGE__
To set a global cache system, use the static method enable_cache. This method receives a hash table with options to pass to the Cache::FastMmap constructor.
Example :
Coat::Persistent->enable_cache(
expire_time => '1h',
cache_size => '50m',
share_file => '/var/cache/myapp.cache',
);
It's possible to disable the cache system with the static method disable_cache.
See Cache::FastMmap for details about available constructor's options.
METHODS
CLASS CONFIGURATION
The following pragma are provided to configure the mapping that will be done between a table and the class.
- has_p $name => %options
-
Coat::Persistent classes have the keyword has_p to define persistent attributes. Attributes declared with has_p are valid Coat attributes and take the same options as Coat's has method. (Refer to Coat for details).
All attributes declared with has_p must exist in the mapped data backend (they are a column of the table mapped to the class).
- has_one $class
-
Tells that current class owns a subobject of the class $class. This will allow you to set and get a subobject transparently.
The backend must have a foreign key to the table of $class.
Example:
package Foo; use Coat::Persistent; has_one 'Bar'; package Bar; use Coat::Persistent; my $foo = new Foo; $foo->bar(new Bar);
- has_many $class
-
This is the same as has_one but says that many items are bound to one instance of the current class. The backend of class $class must provide a foreign key to the current class.
CLASS METHODS
The following methods are inherited by Coat::Persistent classes, they provide features for accessing and touching the database below the abstraction layer. Those methods must be called in class-context.
- find( @conditions )
-
Find operates with three different retrieval approaches:
- Find by id: This can either be a specific id or a list of ids (1, 5, 6)
- Find in scalar context: This will return the first record matched by the options used. These options can either be specific conditions or merely an order. If no record can be matched, undef is returned.
- Find in list context: This will return all the records matched by the options used. If no records are found, an empty array is returned.
-
Example:
my $obj = Class->find(23); my @list = Class->find(1, 23, 34, 54); my $obj = Class->find("field = 'value'"); my $obj = Class->find(["field = ?", $value]);
- find_by_sql($sql, @bind_values
-
Executes a custom sql query against your database and returns all the results if in list context, only the first one if in scalar context.
If you call a complicated SQL query which spans multiple tables the columns specified by the SELECT that aren't real attributes of your model will be provided in the hashref of the object, but you won't have accessors.
The sql parameter is a full sql query as a string. It will be called as is, there will be no database agnostic conversions performed. This should be a last resort because using, for example, MySQL specific terms will lock you to using that particular database engine or require you to change your call if you switch engines.
Example:
my $obj = Class->find_by_sql("select * from class where $cond"); my @obj = Class->find_by_sql("select * from class where col = ?", 34);
- create
-
Creates an object (or multiple objects) and saves it to the database.
The attributes parameter can be either be a hash or an array of hash-refs. These hashes describe the attributes on the objects that are to be created.
Examples
# Create a single new object User->create(first_name => 'Jamie') # Create an Array of new objects User->create([{ first_name => 'Jamie'}, { first_name => 'Jeremy' }])
INSTANCE METHODS
The following methods are provided by objects created from the class. Those methods must be called in instance-context.
- save
-
If no record exists, creates a new record with values matching those of the object attributes. If a record does exist, updates the record with values matching those of the object attributes.
Returns the id of the object saved.
SEE ALSO
See Coat for all the meta-class documentation. See Cache::FastMmap for details about the cache objects provided.
AUTHOR
This module was written by Alexis Sukrieh <sukria@cpan.org>. Quite everything implemented in this module was inspired from ActiveRecord::Base's API (from Ruby on Rails).
Parts of the documentation are also taken from ActiveRecord::Base when appropriate.
COPYRIGHT AND LICENSE
Copyright 2007 by Alexis Sukrieh.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.