ORM Layer

Once you start building complex applications you get the need to abstract your business logic a bit more. Dealing with database queries even with the Query Builder can be tiresome and ugly.

An ORM layer allows you to abstract database entities to models that are more manageable and allow you to create more expressive and readable code, for example, if you have an user table you may do something like this to retrieve a specific user:

use Aurora\Database\Database;

...

$database = resolve(Database::class);
$user = $database->query("SELECT * FROM user WHERE id = ?", [1]);

Or with the Query Builder:

$user = query('user')->where('id', 1)->first();

For a simple query is not that problematic, but what if you wanted to change the status of the user?

$user = query('user')->where('id', 1)->first();
if ($user) {
    query('user')
        ->where('id', 1)
        ->update(['status' => 'Active']);
}

The ORM layer works by creating a Model that encapsulates the Query Builder logic for its specific table and allows you yo do this:

use App\Database\Models\User;

...

$user = User::get(1);
if ($user) {
    $user->status = 'Active';
    $user->save();
}

It's almost the same number of lines, but is more explicit and makes your code more readable and easier to understand.

Creating Models

To create a model from the CLI use the create:model command:

php aurora create:model User

That instructs Aurora to create an User.php file in your app/Database/Models directory with the following structure:

<?php

declare(strict_types = 1);

namespace App\Database\Models;

use Aurora\Database\Model\AbstractModel;

class User extends AbstractModel {

    /**
     * Table name
     * @var string
     */
    protected static $table  = 'user';

    /**
     * Model name
     * @var string
     */
    protected static $model  = 'User';

    /**
     * Model fields
     * @var array
     */
    protected static $fields = [];

    /**
     * Model fields for update
     * @var array
     */
    protected static $update = [];

    /**
     * Default values
     * @var array
     */
    protected static $defaults = [];
}

To get started you'll need to add the model fields; to do so change the $fields and $update properties: put all the table fields on the first and only the ones that should be updated on the latter, for example:

/**
 * Model fields
 * @var array
 */
protected static $fields = [
    'id',
    'login',
    'email',
    'nicename',
    'password',
    'status',
    'type',
    'created',
    'modified'
];

/**
 * Model fields for update
 * @var array
 */
protected static $update = [
    'login',
    'email',
    'nicename',
    'password',
    'status',
    'type',
    'modified'
];

As you can see, we've excluded id and created from the $update fields, so that these ones won't change once the row has been added to the table.

You may also set default values for some or all of the fields:

/**
 * Default values
 * @var array
 */
protected static $defaults = [
    'type' => 'Subscriber',
    'status' => 'Inactive'
];

Fetching models

Fetching models is really easy, you can for example to get all the users you can do:

$users = User::all();

To get a single user by its ID, use the get() method:

$user = User::get(1);

Also the Model wraps a Query object, so you can apply where filters and even sorting/paging for example:

$users = User::where()
    ->where('status', 'Active')
    ->sort('created', 'asc')
    ->page(3, 15)
    ->all();

Saving and updating models

Creating a model from scratch is straightforward, just create a variable, set its fields and call the save() method:

$user = new User();
$user->login = 'foo';
$user->nicename = 'Foo';
$user->email = '[email protected]';
$user->type = 'Subscriber';
$user->password = '53cr3tP455';
$user->save();

Updating is even easier, for example, to change the nicename field you can simply:

$user = User::get(1);
$user->nicename = 'Chuck Norris';
$user->save();

The save() method updates the row on the table if it exists and creates it if it does nor, that's called an upsert.

Next up, Migrations.