cjmellor / approval
Approve or Deny new Model data before it is persisted
Installs: 30 173
Dependents: 0
Suggesters: 0
Security: 0
Stars: 364
Watchers: 3
Forks: 24
Open Issues: 1
pkg:composer/cjmellor/approval
Requires
- php: ^8.2
- illuminate/contracts: ^10.0|^11.0|^12.0
- spatie/laravel-package-tools: ^1.18
Requires (Dev)
- laravel/pint: ^1.0
- nunomaduro/collision: ^7.0|^8.0
- orchestra/testbench: ^7.0|^8.0|^9.0|^10.0
- pestphp/pest: ^2.0|^3.7
- pestphp/pest-plugin-arch: ^2.0|^3.0
- pestphp/pest-plugin-laravel: ^2.0|^3.1
- pestphp/pest-plugin-type-coverage: ^2.0|^3.3
- 2.x-dev
- v1.6.6
- v1.6.5
- v1.6.4
- v1.6.3
- v1.6.2
- v1.6.1
- v1.6.0
- v1.5.0
- v1.4.5
- v1.4.4
- v1.4.3
- v1.4.2
- v1.4.1
- v1.4.0
- v1.3.1
- v1.3.0
- v1.2.0
- v1.1.5
- v1.1.4
- v1.1.3
- v1.1.2
- v1.1.1
- v1.1.0
- v1.0.1
- 1.0
- dev-dependabot/github_actions/stefanzweifel/git-auto-commit-action-7
- dev-dependabot/github_actions/actions/checkout-5
- dev-main
This package is auto-updated.
Last update: 2025-10-15 08:20:38 UTC
README
Approval is a Laravel package that provides a simple way to approve new Model data before it is persisted.
Installation
You can install the package via composer:
composer require cjmellor/approval
You can publish and run the migrations with:
php artisan vendor:publish --tag="approval-migrations"
php artisan migrate
Upgrading from v1
If you're upgrading from v1.x to v2.x, please follow the detailed upgrade guide to ensure a smooth transition. Version 2 introduces database schema changes that require running specific commands in the correct order.
You can publish the config file with:
php artisan vendor:publish --tag="approval-config"
This is the contents of the published config file:
return [ 'approval' => [ /** * The approval polymorphic pivot name * * Default: 'approvalable' */ 'approval_pivot' => 'approvalable', ], ];
The config allows you to change the polymorphic pivot name. It should end with able
though.
Usage
Note
This package does not approve/deny the data for you, it just stores the new/amended data into the database. It is up to you to decide how you implement a function to approve or deny the Model.
Add the MustBeApproved
trait to your Model and now the data will be stored in an approvals
table, ready for you to approve or deny.
For example, you add it to a Post
Model and each time a Post is created or updated, all the dirty data will be stored in the database as JSON for you to do something with it.
<?php use Cjmellor\Approval\Concerns\MustBeApproved; class Post extends Model { use MustBeApproved; // ... }
All Models using the Trait will now be stored in a new table -- approvals
. This is a polymorphic relationship.
Here is some info about the columns in the approvals
table:
approvalable_type
=> The class name of the Model that the approval is for
approvalable_id
=> The ID of the Model that the approval is for
state
=> The state of the approval. This uses an Enum class. This column is cast to an ApprovalStatus
Enum class
new_data
=> All the fields created or updated in the Model. This is a JSON column. This column is cast to the AsArrayObject
Cast
original_data
=> All the fields in the Model before they were updated. This is a JSON column. This column is cast to the AsArrayObject
Cast
rolled_back_at
=> A timestamp of when this was last rolled back to its original state
audited_at
=> The ID of the User who set the state
foreign_key
=> A foreign key to the Model that the approval is for
creator_id
=> The ID of the model who requested the approval
creator_type
=> The class name of the model who requested the approval
Bypassing Approval Check
If you want to check if the Model data will be bypassed, use the isApprovalBypassed
method.
return $model->isApprovalBypassed();
Foreign Keys for New Models
Note
It is recommended to read the below section on how foreign keys work in this package.
Important
By default, the foreign key will always be user_id
because this is the most common foreign key used in Laravel.
If you create a new Model directly via the Model, e.g.
Post::create(['title' => 'Some Title']);
be sure to also add the foreign key to the Model, e.g.
Post::create(['title' => 'Some Title', 'user_id' => 1]);
Now when the Model is sent for approval, the foreign key will be stored in the foreign_key
column.
Customise the Foreign Key
Your Model might not use the user_id
as the foreign key, so you can customise it by adding this method to your Model:
public function getApprovalForeignKeyName(): string { return 'author_id'; }
Scopes
The package comes with some helper methods for the Builder, utilising a custom scope - ApprovalStateScope
By default, all queries to the approvals
table will return all the Models' no matter the state.
There are three methods to help you retrieve the state of the Approval.
<?php use App\Models\Approval; Approval::approved()->get(); Approval::rejected()->get(); Approval::pending()->count();
You can also set a state for an approval:
<?php use App\Models\Approval; Approval::where('id', 1)->approve(); Approval::where('id', 2)->reject(); Approval::where('id', 3)->postpone();
In the event you need to reset a state, you can use the withAnyState
helper.
Helpers
Conditional helper methods are used, so you can set the state of an Approval when a condition is met.
$approval->approveIf(true); $approval->rejectIf(false); $approval->postponeIf(true); $approval->approveUnless(false); $approval->rejectUnless(true); $approval->postponeUnless(false);
Requestor Functionality
The package includes methods to work with the creator/requestor of an approval:
// Get the requestor (creator) of the approval $requestor = $approval->requestor;
// Filter approvals by requestor $userApprovals = Approval::requestedBy($user)->get();
// Check if an approval was requested by a specific user if ($approval->wasRequestedBy($user)) { // Do something }
Events
Once a Model's state has been changed, an event will be fired.
- ModelApproved::class - ModelPostponed::class - ModelRejected::class - ApprovalCreated::class
Configurable Approval States
The package allows you to define custom approval states beyond the default set (Pending
, Approved
, Rejected
).
Configuring Custom States
Define your custom states in the config/approval.php
file:
'states' => [ 'pending' => [ 'name' => 'Pending', 'default' => true, ], 'approved' => [ 'name' => 'Approved', ], 'rejected' => [ 'name' => 'Rejected', ], 'in_review' => [ 'name' => 'In Review', ], 'needs_info' => [ 'name' => 'Needs Clarification', ], ],
Using Custom States
You can set any configured state on an approval:
// Set a custom state $approval->setState('in_review'); // Check the current state $currentState = $approval->getState();
Querying by State
The package provides a flexible way to query approvals by any state:
// Query approvals with a specific state $inReviewApprovals = Approval::whereState('in_review')->get(); // The standard scopes still work for the default states $pendingApprovals = Approval::pending()->get();
Standard states (pending
, approved
, rejected
) continue to work with all existing methods, ensuring backward compatibility.
Rollbacks
If you need to roll back an approval, you can use the rollback
method.
Note
By default, a Rollback will bypass been added back to the approvals
table
Approval::first()->rollback();
This will revert the data and set the state to pending
and touch the rolled_back_at
timestamp, so you have a record of when it was rolled back.
If you want a Rollback to be re-approved, pass the bypass
parameter as false
to the rollback
method
Approval::first()->rollback(bypass: false); // default is true
Conditional Rollbacks
A roll-back can be conditional, so you can roll back an approval if a condition is met.
Approval::first()->rollback(fn () => true);
Events
When a Model has been rolled back, a ModelRolledBack
event will be fired with the Approval Model that was rolled back, as well as the User that rolled it back.
// ModelRolledBackEvent::class public Model $approval, public Authenticatable|null $user,
Time-Based Approvals
The package supports automatic actions for approvals that aren't completed within a set time frame.
Setting Expiration Times
You can set an expiration time on any approval:
// Set expiration in hours (most common) Approval::find(1)->expiresIn(hours: 24); // Set expiration in minutes Approval::find(1)->expiresIn(minutes: 30); // Set expiration in days Approval::find(1)->expiresIn(days: 7); // Set specific expiration datetime Approval::find(1)->expiresIn(datetime: now()->addWeek());
Automatic Actions
You can define what happens when an approval expires:
// Automatically reject when expired Approval::find(1)->expiresIn(hours: 48)->thenReject(); // Automatically postpone (set to pending) when expired Approval::find(1)->expiresIn(hours: 48)->thenPostpone(); // Use a custom action through event listeners Approval::find(1)->expiresIn(hours: 48)->thenDo(function($approval) { // This callback is for documentation only // Implement an event listener for ApprovalExpired event });
Processing Expired Approvals
To process expired approvals, add this command to your scheduler:
// In App\Console\Kernel.php protected function schedule(Schedule $schedule) { $schedule->command('approval:process-expired')->everyMinute(); }
Querying Expirations
You can query approvals based on their expiration status:
// Get all expired approvals Approval::expired()->get(); // Get all non-expired approvals (including those with no expiration) Approval::notExpired()->get(); // Get all approvals that have an expiration set Approval::hasExpiration()->get(); // Check if a specific approval is expired $approval->isExpired();
Events
When an approval expires and is processed, these events are fired:
ApprovalExpired
: Fired for all expired approvals- Followed by the specific action event (
ModelRejected
,ModelSetPending
, etc.)
Disable Approvals
If you don't want Model data to be approved, you can bypass it with the withoutApproval
method.
$model->withoutApproval()->update(['title' => 'Some Title']);
Specify Approvable Attributes
By default, all attributes of the model will go through the approval process, however if you only wish certain attributes to go through this process, you can specify them using the approvalAttributes
property in your model.
<?php use Cjmellor\Approval\Concerns\MustBeApproved; class Post extends Model { use MustBeApproved; protected array $approvalAttributes = ['name']; // ... }
In this example, only the name attribute of this model will go through the approval process, all mutations on other attributes will bypass the approval process.
If you omit the approvalAttributes
property from your model, all attributes will go through the approval process.
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please open a PR with as much detail as possible about what you're trying to achieve.
Credits
License
The MIT Licence (MIT). Please see Licence File for more information.