Database: Models (Active Record ORM)
Introduction
Winter provides a beautiful and simple Active Record implementation for working with your database, based on Eloquent by Laravel. Each database table has a corresponding "Model" which is used to interact with that table. Models allow you to query for data in your tables, as well as insert new records into the table.
Model classes reside in the models subdirectory of a plugin directory. An example of a model directory structure:
📂 plugins
┗ 📂 acme
┗ 📂 blog
┣ 📂 models
┃ ┣ 📂 user <=== Model config directory
┃ ┃ ┣ 📜 columns.yaml <=== Model config files
┃ ┃ ┗ 📜 fields.yaml <==^
┃ ┗ 📜 User.php <=== Model class
┗ 📜 Plugin.php
The model configuration directory could contain the model's list column and form field definitions. The model configuration directory name matches the model class name written in lowercase.
Defining models
In most cases, you should create one model class for each database table. All model classes must extend the Model
class. The most basic representation of a model used inside a Plugin looks like this:
namespace Acme\Blog\Models;
use Model;
class Post extends Model
{
/**
* The table associated with the model.
*
* @var string
*/
protected $table = 'acme_blog_posts';
}
The $table
protected field specifies the database table corresponding the model. The table name is a snake case name of the author, plugin and pluralized record type name.
Supported properties
There are some standard properties that can be found on models, in addition to those provided by model traits. For example:
class User extends Model
{
protected $primaryKey = 'id';
public $exists = false;
protected $dates = ['last_seen_at'];
public $timestamps = true;
protected $jsonable = ['permissions'];
protected $guarded = ['*'];
}
Property | Description |
---|---|
$primaryKey |
primary key name used to identify the model. |
$incrementing |
boolean that if false indicates that the primary key is not an incrementing integer value. |
$exists |
boolean that if true indicates that the model exists. |
$dates |
values are converted to an instance of Carbon/DateTime objects after fetching. |
$timestamps |
boolean that if true will automatically set created_at and updated_at fields. |
$jsonable |
values are encoded as JSON before saving and converted to arrays after fetching. |
$fillable |
values are fields accessible to mass assignment. |
$guarded |
values are fields guarded from mass assignment. |
$visible |
values are fields made visible when serializing the model data. |
$hidden |
values are fields made hidden when serializing the model data. |
$connection |
string that contains the connection name that's utilised by the model by default. |
Primary key
Models will assume that each table has a primary key column named id
. You may define a $primaryKey
property to override this convention.
class Post extends Model
{
/**
* The primary key for the model.
*
* @var string
*/
protected $primaryKey = 'id';
}
Incrementing
Models will assume that the primary key is an incrementing integer value, which means that by default the primary key will be cast to an integer automatically. If you wish to use a non-incrementing or a non-numeric primary key you must set the public $incrementing
property to false.
class Message extends Model
{
/**
* The primary key for the model is not an integer.
*
* @var bool
*/
public $incrementing = false;
}
Timestamps
By default, a model will expect created_at
and updated_at
columns to exist on your tables. If you do not wish to have these columns managed automatically, set the $timestamps
property on your model to false
:
class Post extends Model
{
/**
* Indicates if the model should be timestamped.
*
* @var bool
*/
public $timestamps = false;
}
If you need to customize the format of your timestamps, set the $dateFormat
property on your model. This property determines how date attributes are stored in the database, as well as their format when the model is serialized to an array or JSON:
class Post extends Model
{
/**
* The storage format of the model's date columns.
*
* @var string
*/
protected $dateFormat = 'U';
}
Values stored as JSON
When attributes names are passed to the $jsonable
property, the values will be serialized and deserialized from the database as JSON:
class Post extends Model
{
/**
* @var array Attribute names to encode and decode using JSON.
*/
protected $jsonable = ['data'];
}
Retrieving models
When requesting data from the database the model will retrieve values primarily using the get
or first
methods, depending on whether you wish to retrieve multiple models or retrieve a single model respectively. Queries that derive from a Model return an instance of Winter\Storm\Database\Builder.
NOTE: All model queries have in-memory caching enabled by default. While the cache should automatically invalidate itself most of the time, sometimes you will need to use the
$model->reload()
method to flush the cache for more complex use cases.
Retrieving multiple models
Once you have created a model and its associated database table, you are ready to start retrieving data from your database. Think of each model as a powerful query builder allowing you to query the database table associated with the model. For example:
$flights = Flight::all();
Accessing column values
If you have a model instance, you may access the column values of the model by accessing the corresponding property. For example, let's loop through each Flight
instance returned by our query and echo the value of the name
column:
foreach ($flights as $flight) {
echo $flight->name;
}
Adding additional constraints
The all
method will return all of the results in the model's table. Since each model serves as a query builder, you may also add constraints to queries, and then use the get
method to retrieve the results:
$flights = Flight::where('active', 1)
->orderBy('name', 'desc')
->take(10)
->get();
NOTE: Since models are query builders, you should familiarize yourself with all of the methods available on the query builder. You may use any of these methods in your model queries.
Collections
For methods like all
and get
which retrieve multiple results, an instance of a Collection
will be returned. This class provides a variety of helpful methods for working with your results. Of course, you can simply loop over this collection like an array:
foreach ($flights as $flight) {
echo $flight->name;
}
Chunking results
If you need to process thousands of records, use the chunk
command. The chunk
method will retrieve a "chunk" of models, feeding them to a given Closure
for processing. Using the chunk
method will conserve memory when working with large result sets:
Flight::chunk(200, function ($flights) {
foreach ($flights as $flight) {
//
}
});
The first argument passed to the method is the number of records you wish to receive per "chunk". The Closure passed as the second argument will be called for each chunk that is retrieved from the database.
Retrieving a single model
In addition to retrieving all of the records for a given table, you may also retrieve single records using find
and first
. Instead of returning a collection of models, these methods return a single model instance:
// Retrieve a model by its primary key
$flight = Flight::find(1);
// Retrieve the first model matching the query constraints
$flight = Flight::where('active', 1)->first();
Not found exceptions
Sometimes you may wish to throw an exception if a model is not found. This is particularly useful in routes or controllers. The findOrFail
and firstOrFail
methods will retrieve the first result of the query. However, if no result is found, a Illuminate\Database\Eloquent\ModelNotFoundException
will be thrown:
$model = Flight::findOrFail(1);
$model = Flight::where('legs', '>', 100)->firstOrFail();
When developing an API, if the exception is not caught, a 404
HTTP response is automatically sent back to the user, so it is not necessary to write explicit checks to return 404
responses when using these methods:
Route::get('/api/flights/{id}', function ($id) {
return Flight::findOrFail($id);
});
Retrieving aggregates
You may also use count
, sum
, max
, and other aggregate functions provided by the query builder. These methods return the appropriate scalar value instead of a full model instance:
$count = Flight::where('active', 1)->count();
$max = Flight::where('active', 1)->max('price');
Inserting & updating models
Inserting and updating data are the cornerstone feature of models, it makes the process effortless when compared to traditional SQL statements.
Basic inserts
To create a new record in the database, simply create a new model instance, set attributes on the model, then call the save
method:
$flight = new Flight;
$flight->name = 'Sydney to Canberra';
$flight->save();
In this example, we simply create a new instance of the Flight
model and assign the name
attribute. When we call the save
method, a record will be inserted into the database. The created_at
and updated_at
timestamps will automatically be set too, so there is no need to set them manually.
Basic updates
The save
method may also be used to update models that already exist in the database. To update a model, you should retrieve it, set any attributes you wish to update, and then call the save
method. Again, the updated_at
timestamp will automatically be updated, so there is no need to manually set its value:
$flight = Flight::find(1);
$flight->name = 'Darwin to Adelaide';
$flight->save();
Updates can also be performed against any number of models that match a given query. In this example, all flights that are active
and have a destination
of San Diego
will be marked as delayed:
Flight::where('is_active', true)
->where('destination', 'Perth')
->update(['delayed' => true]);
The update
method expects an array of column and value pairs representing the columns that should be updated.
Update or Insert / upsert()
(Batch query to process multiple rows in one DB call)
If you would like to perform multiple "upserts" in a single query, then you should use the upsert
method instead. The method's first argument consists of the values to insert or update, while the second argument lists the column(s) that uniquely identify records within the associated table. The method's third and final argument is an array of the columns that should be updated if a matching record already exists in the database. The upsert
method will automatically set the created_at
and updated_at
timestamps if timestamps are enabled on the model:
MyVendor\MyPlugin\Models\Flight::upsert([
['departure' => 'Oakland', 'destination' => 'San Diego', 'price' => 99],
['departure' => 'Chicago', 'destination' => 'New York', 'price' => 150]
], ['departure', 'destination'], ['price']);
NOTE:: All databases except SQL Server require the columns in the second argument of the
upsert
method to have a "primary" or "unique" index.
Mass assignment
You may also use the create
method to save a new model in a single line. The inserted model instance will be returned to you from the method. However, before doing so, you will need to specify either a fillable
or guarded
attribute on the model, as all models protect against mass-assignment by default. Note that neither fillable
or guarded
affect the submission of backend forms, only the use of create
or fill
method directly on the model.
A mass-assignment vulnerability occurs when a user passes an unexpected HTTP parameter through a request, and that parameter changes a column in your database you did not expect. For example, a malicious user might send an is_admin
parameter through an HTTP request, which is then mapped onto your model's create
method, allowing the user to escalate themselves to an administrator.
To get started, you should define which model attributes you want to make mass assignable. You may do this using the $fillable
property on the model. For example, let's make the name
attribute of our Flight
model mass assignable:
class Flight extends Model
{
/**
* The attributes that are mass assignable.
*
* @var array
*/
protected $fillable = ['name'];
}
Once we have made the attributes mass assignable, we can use the create
method to insert a new record in the database. The create
method returns the saved model instance:
$flight = Flight::create(['name' => 'Flight 10']);
Using the $fillable
property makes it clear which fields are allowed to be mass-assignable, but does require the
developer to add additional fields to this array when new columns are added to the database. Conversely, you may use the $guarded
property instead to assume all fields are fillable except for the fields defined in the $guarded
array. This can be useful for models with large amounts of fields.
class Flight extends Model
{
/**
* The attributes that aren't mass assignable.
*
* @var array
*/
protected $guarded = ['price'];
}
In the example above, all attributes except for price
will be mass-assignable.
TIP: In general, we recommend the use of
$fillable
over$guarded
, as does the Laravel framework. While it does involve an extra step when creating new fields for your models, it also guards against potentially unwanted or malicious data being written to new fields through prior functionality.
Other creation methods
Sometimes you may wish to only instantiate a new instance of a model. You can do this using the make
method. The make
method will simply return a new instance without saving or creating anything.
$flight = Flight::make(['name' => 'Flight 10']);
// Functionally the same as...
$flight = new Flight;
$flight->fill(['name' => 'Flight 10']);
There are two other methods you may use to create models by mass assigning attributes: firstOrCreate
and firstOrNew
. The firstOrCreate
method will attempt to locate a database record using the given column / value pairs. If the model can not be found in the database, a record will be inserted with the given attributes.
The firstOrNew
method, like firstOrCreate
will attempt to locate a record in the database matching the given attributes. However, if a model is not found, a new model instance will be returned. Note that the model returned by firstOrNew
has not yet been persisted to the database. You will need to call save
manually to persist it:
// Retrieve the flight by the attributes, otherwise create it
$flight = Flight::firstOrCreate(['name' => 'Flight 10']);
// Retrieve the flight by the attributes, or instantiate a new instance
$flight = Flight::firstOrNew(['name' => 'Flight 10']);
Deleting models
To delete a model, call the delete
method on a model instance:
$flight = Flight::find(1);
$flight->delete();
Deleting an existing model by key
In the example above, we are retrieving the model from the database before calling the delete
method. However, if you know the primary key of the model, you may delete the model without retrieving it. To do so, call the destroy
method:
Flight::destroy(1);
Flight::destroy([1, 2, 3]);
Flight::destroy(1, 2, 3);
Deleting models by query
You may also run a delete query on a set of models. In this example, we will delete all flights that are marked as inactive:
$deletedRows = Flight::where('active', 0)->delete();
NOTE: It is important to mention that model events will not fire when deleting records directly from a query.
Query scopes
Local scopes
Scopes allow you to define common sets of constraints that you may easily re-use throughout your application. For example, you may need to frequently retrieve all users that are considered "popular". To define a scope, simply prefix a model method with scope
:
class User extends Model
{
/**
* Scope a query to only include popular users.
*/
public function scopePopular($query)
{
return $query->where('votes', '>', 100);
}
/**
* Scope a query to only include active users.
*/
public function scopeActive($query)
{
return $query->where('is_active', 1);
}
}
Utilizing a query scope
Once the scope has been defined, you may call the scope methods when querying the model. However, you do not need to include the scope
prefix when calling the method. You can even chain calls to various scopes, for example:
$users = User::popular()->active()->orderBy('created_at')->get();
Dynamic scopes
Sometimes you may wish to define a scope that accepts parameters. To get started, just add your additional parameters to your scope. Scope parameters should be defined after the $query
argument:
class User extends Model
{
/**
* Scope a query to only include users of a given type.
*/
public function scopeApplyType($query, $type)
{
return $query->where('type', $type);
}
}
Now you may pass the parameters when calling the scope:
$users = User::applyType('admin')->get();
Global scopes
Global scopes allow you to add constraints to all queries for a given model. Winters own soft delete functionality utilizes global scopes to only retrieve "non-deleted" models from the database. Writing your own global scopes can provide a convenient, easy way to make sure every query for a given model receives certain constraints.
Writing Global Scopes
Writing a global scope is simple. First, define a class that implements the Illuminate\Database\Eloquent\Scope
interface. Winter does not have a conventional location that you should place scope classes, so you are free to place this class in any directory that you wish.
The Scope
interface requires you to implement one method: apply
. The apply
method may add where
constraints or other types of clauses to the query as needed:
<?php
namespace MyAuthor\MyPlugin\Scopes;
use Winter\Storm\Database\Builder;
use Winter\Storm\Database\Model;
use Illuminate\Database\Eloquent\Scope;
class AncientScope implements Scope
{
/**
* Apply the scope to a given Eloquent query builder.
*
* @param Winter\Storm\Database\Builder $builder
* @param Winter\Storm\Database\Model $model
* @return void
*/
public function apply(Builder $builder, Model $model)
{
$builder->where('created_at', '<', now()->subYears(2000));
}
}
NOTE: If your global scope is adding columns to the select clause of the query, you should use the addSelect method instead of select. This will prevent the unintentional replacement of the query's existing select clause.
Applying Global Scopes
To assign a global scope to a model, you should override the model's booted
method and invoke the model's addGlobalScope
method. The addGlobalScope
method accepts an instance of your scope as its only argument:
<?php
namespace MyAuthor\MyPlugin\Models;
use MyAuthor\MyPlugin\Scopes\AncientScope;
use Winter\Storm\Database\Model;
class User extends Model
{
/**
* The "booted" method of the model.
*
* @return void
*/
protected static function booted()
{
static::addGlobalScope(new AncientScope);
}
}
After adding the scope in the example above to the MyAuthor\MyPlugin\Models\User
model, a call to the User::all()
method will execute the following SQL query:
select * from `users` where `created_at` < 0021-02-18 00:00:00
Anonymous Global Scopes
Winter also allows you to define global scopes using closures, which is particularly useful for simple scopes that do not warrant a separate class of their own. When defining a global scope using a closure, you should provide a scope name of your own choosing as the first argument to the addGlobalScope method:
<?php
namespace MyAuthor\MyPlugin\Models;
use Winter\Storm\Database\Builder;
use Winter\Storm\Database\Model;
class User extends Model
{
/**
* The "booted" method of the model.
*
* @return void
*/
protected static function booted()
{
static::addGlobalScope('ancient', function (Builder $builder) {
$builder->where('created_at', '<', now()->subYears(2000));
});
}
}
Removing Global Scopes
If you would like to remove a global scope for a given query, you may use the withoutGlobalScope
method. This method accepts the class name of the global scope as its only argument:
User::withoutGlobalScope(AncientScope::class)->get();
Or, if you defined the global scope using a closure, you should pass the string name that you assigned to the global scope:
User::withoutGlobalScope('ancient')->get();
If you would like to remove several or even all of the query's global scopes, you may use the withoutGlobalScopes
method:
// Remove all of the global scopes...
User::withoutGlobalScopes()->get();
// Remove some of the global scopes...
User::withoutGlobalScopes([
FirstScope::class, SecondScope::class
])->get();
Events
Models fire several events, allowing you to hook into various points in the model's lifecycle. Events allow you to easily execute code each time a specific model class is saved or updated in the database. Events are defined by overriding special methods in the class, the following method overrides are available:
Event | Description |
---|---|
beforeCreate |
before the model is saved, when first created. |
afterCreate |
after the model is saved, when first created. |
beforeSave |
before the model is saved, either created or updated. |
afterSave |
after the model is saved, either created or updated. |
beforeValidate |
before the supplied model data is validated. |
afterValidate |
after the supplied model data has been validated. |
beforeUpdate |
before an existing model is saved. |
afterUpdate |
after an existing model is saved. |
beforeDelete |
before an existing model is deleted. |
afterDelete |
after an existing model is deleted. |
beforeRestore |
before a soft-deleted model is restored. |
afterRestore |
after a soft-deleted model has been restored. |
beforeFetch |
before an existing model is populated. |
afterFetch |
after an existing model has been populated. |
An example of using an event:
/**
* Generate a URL slug for this model
*/
public function beforeCreate()
{
$this->slug = Str::slug($this->name);
}
NOTE: Relationships created with deferred-binding (i.e: file attachments) will not be available in the
afterSave
model event if they have not been committed yet. To access uncommitted bindings, use thewithDeferred($sessionKey)
method on the relation. Example:$this->images()->withDeferred(post('_session_key'))->get();
Basic usage
Whenever a new model is saved for the first time, the beforeCreate
and afterCreate
events will fire. If a model already existed in the database and the save
method is called, the beforeUpdate
/ afterUpdate
events will fire. However, in both cases, the beforeSave
/ afterSave
events will fire.
For example, let's define an event listener that populates the slug attribute when a model is first created:
/**
* Generate a URL slug for this model
*/
public function beforeCreate()
{
$this->slug = Str::slug($this->name);
}
Returning false
from an event will cancel the save
/ update
operation:
public function beforeCreate()
{
if (!$user->isValid()) {
return false;
}
}
It's possible to access old values using the original
attribute. For example:
public function afterUpdate()
{
if ($this->title != $this->original['title']) {
// title changed
}
}
You can externally bind to local events for a single instance of a model using the bindEvent
method. The event name should be the same as the method override name, prefixed with model.
.
$flight = new Flight;
$flight->bindEvent('model.beforeCreate', function() use ($model) {
$model->slug = Str::slug($model->name);
})
Extending models
Since models are equipped to use behaviors, they can be extended with the static extend
method. The method takes a closure and passes the model object into it.
Inside the closure you can add relations to the model. Here we extend the Backend\Models\User
model to include a profile (has one) relationship referencing the Acme\Demo\Models\Profile
model.
\Backend\Models\User::extend(function($model) {
$model->hasOne['profile'] = ['Acme\Demo\Models\Profile', 'key' => 'user_id'];
});
This approach can also be used to bind to local events, the following code listens for the model.beforeSave
event.
\Backend\Models\User::extend(function($model) {
$model->bindEvent('model.beforeSave', function() use ($model) {
// ...
});
});
NOTE: Typically the best place to place code is within your plugin registration class
boot
method as this will be run on every request ensuring that the extensions you make to the model are available everywhere.
Property methods
Additionally, a few methods exist to extend protected model properties.
\Backend\Models\User::extend(function($model) {
// add cast attributes
$model->addCasts([
'some_extended_field' => 'int',
]);
// add a date attribute
$model->addDateAttribute('updated_at');
// add fillable or jsonable fields
// these methods accept one or more strings, or an array of strings
$model->addFillable('first_name');
$model->addJsonable('some_data');
});
Relationships methods
The following methods are provided by the HasRelationship trait:
public function addHasOneRelation(string $name, array $config)
public function addHasManyRelation(string $name, array $config)
public function addBelongsToRelation(string $name, array $config)
public function addBelongsToManyRelation(string $name, array $config)
public function addMorphToRelation(string $name, array $config)
public function addMorphOneRelation(string $name, array $config)
public function addMorphManyRelation(string $name, array $config)
public function addMorphToManyRelation(string $name, array $config)
public function addMorphedByManyRelation(string $name, array $config)
public function addAttachOneRelation(string $name, array $config)
public function addAttachManyRelation(string $name, array $config)
public function addHasOneThroughRelation(string $name, array $config)
public function addHasManyThroughRelation(string $name, array $config)
It is strongly suggested to use the above methods to add relations when extending a model since they will merge the existing relations and make sure the relation is valid and does not already exist.
Example usage:
\Backend\Models\User::extend(function($model) {
$model->addHasOne('profile', ['Acme\Demo\Models\Profile', 'key' => 'user_id']);
});