orm
is a small and simple database layer implementing the active record pattern. Its aim is to be simple, fast and useful. It relies on the [clancats hydrahon] query builder for database query access.
- PDO underneath so should work with any PDO compatible database
- Provides a full query builder interface using clancats/hydrahon
- Supports full save, destroy, update, insert functionality through the active record implementation
- Supports model finder objects to help avoid 'fat model' syndrome
- Supports fine grained model hooks for precise control of model lifecycle data
- Supports model validation using respect/validation
- Simple relationship handling - belongsTo, hasMany, hasOne
- Configurable data type conversion for model properties
- Migrations - you can use whatever you like to manage your schema. We recommend phinx
- Multiple database connections - currently
orm
only supports a single PDO connection.
You'll need at least PHP7.0 to use the library. The recommended way to install is using composer:
composer install ronanchilvers/orm
Since orm
uses PDO, it's up to you how you create and instantiate your PDO object. This
will probably be in your bootstrap somewhere. Once your PDO object is available you
will need to give it to orm
. Here's an example:
$pdo = new PDO('sqlite::memory:');
Ronanchilvers\Orm\Orm::setConnection($pdo);
Clearly you will almost certainly not be using a :memory:
DSN in practice. However
you create your PDO object, the crucial point here is that you call Orm::setConnection
to provide orm
with your connection object.
orm
doesn't make any pre-judgments about your database schema. When building models
from a database it assumes it maps columns to properties and when saving it assumes
that any property has a corresponding database table column. Its up to you to make
sure the data makes sense.
Here we assume that we have a database table that looks like the following. We're using MySQL / MariaDB syntax here but whatever PDO supports should be fine.
CREATE TABLE `books` (
`book_id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`book_author` int(11) NOT NULL,
`book_name` varchar(1024) NOT NULL DEFAULT '',
`book_created` datetime DEFAULT CURRENT_TIMESTAMP,
`book_updated` datetime DEFAULT NULL,
PRIMARY KEY (`book_id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
First create your model class. It should extend the Ronanchilvers\Orm\Model
class. As with many active record implementations we assume here that the table
has a plural name (books) and its corresponding model class will have a singular
name (book).
class Book extends Model
{}
If your columns have no column prefix, that's all you need to do. The table name will be inferred from the class name. If however, like the example above, you have column prefixes then you can tweak your model to suit.
class Book extends Model
{
static protected $columnPrefix = 'book';
}
Similarly if your table doesn't map to the model name you can specufy that too.
class Book extends Model
{
static protected $table = 'my_books_table';
static protected $columnPrefix = 'book';
}
Now you're ready to use the model.
orm
supports a query builder interface provided by clancats/hydrahon. In order
to retrieve models from the database, first obtain a finder object.
$finder = Orm::finder(Book::class);
Then you can use the finder object to retrieve models.
$books = $finder->all();
There are several standard finder methods you can use:
// Get all the records in one go
$books = $finder->all();
// Get the third page of models when there are 30 records per page
// (10 per page is the default)
$books = $finder->all(3, 30);
// Get a specific model by its primary key, here assumed to be numeric
$book = $finder->one(23);
You can use the full query builder to gain more control over the query:
// Get all the books for author id 20
$books = $finder->select()->where('book_author', 20);
// Get all books added since last week - here we're using the excellent Carbon wrapper
// for DateTime
$recentBooks = $finder->select()->where('book_created', '>', Carbon::now()->subWeek());
You can read more about the capabilities of the query builder over at the clancats/hydrahon site.
If you want complete control over the SQL that is executed you can do:
$sql = "SELECT *
FROM books
LEFT JOIN authors ON author_id = book_author
WHERE author_name LIKE :name
AND author_created < :created";
$params = [
'name' => 'Fred%',
'created' => Carbon::now()->subYear()->format('Y-m-d H:i:s'),
];
$books = $finder->query($sql, $params);
By default Orm::finder
will give you back a vanilla Ronanchilvers\Orm\Finder
object tied to given model class. However you can also override the finder class for a given model by setting the $finder
static property on your model. This can be very useful if you want to add custom find methods (for example).
Let's imagine you've created a BookFinder
class:
class BookFinder extends Finder
{
public function forAuthorId(int $id): array
{
return $this
->select()
->where(Book::prefix('author'), $id)
->execute();
}
}
You can see that we've created a subclass of the default finder and we've added a forAuthorId
method which returns an array of books for a given author id.
Then we tell the model about our new finder class:
class Book extends Model
{
static protected $finder = BookFinder::class;
}
and then finally you're readdy to use it:
// $finder will be an instance of BookFinder here
$finder = Orm::finder(Book::class);
$authors = $finder->forAuthorId(2);
Saving a model back to the database is simply a matter of calling the save
method.
$finder = Orm::finder(Book::class);
$book = $finder->one(23);
$book->save();
Similarly destroying a model is also simple:
$book->delete();
As in any active record implementation, database table columns are expressed as properties on the model instance. Consequently data can be accessed by accessing those properties.
$finder = Orm::finder(Book::class);
$book = $finder->one(72);
echo $book->name;