hasyirin / laravel-kpi
Measure turnaround time against a working schedule, and track status transitions (movements) on any Eloquent model.
Fund package maintenance!
Requires
- php: ^8.4
- ext-bcmath: *
- illuminate/contracts: ^12.0||^13.0
- spatie/laravel-package-tools: ^1.16
Requires (Dev)
- larastan/larastan: ^3.0
- laravel/pint: ^1.14
- nunomaduro/collision: ^8.1.1
- orchestra/testbench: ^10.0||^11.0
- pestphp/pest: ^3.0||^4.0
- pestphp/pest-plugin-arch: ^3.0||^4.0
- pestphp/pest-plugin-laravel: ^3.0||^4.0
- phpstan/extension-installer: ^1.3||^2.0
- phpstan/phpstan-deprecation-rules: ^1.1||^2.0
- phpstan/phpstan-phpunit: ^1.3||^2.0
- spatie/laravel-ray: ^1.35
README
Measure turnaround time against a working schedule, and track status transitions (movements) on any Eloquent model.
Given a start and end timestamp, laravel-kpi computes the effective working duration — skipping weekends, holidays, and any custom exclude dates — expressed as minutes, hours, and a period ratio against scheduled working minutes. It also ships a lightweight workflow layer: attach the InteractsWithMovement trait to a model and you get a chain of status transitions (pass, passIfNotCurrent), each stamped with the KPI duration it took.
Requirements
- PHP 8.4+
- Laravel 12 or 13
ext-bcmath
Installation
composer require hasyirin/laravel-kpi
Publish and run the migrations:
php artisan vendor:publish --tag="laravel-kpi-migrations"
php artisan migrate
Publish the config file:
php artisan vendor:publish --tag="laravel-kpi-config"
Configuration
config/kpi.php:
use Hasyirin\KPI\Enums\Day; use Hasyirin\KPI\Models\Holiday; use Hasyirin\KPI\Models\Movement; use Hasyirin\KPI\Models\RecurringHoliday; return [ 'formats' => [ 'datetime' => 'd/m/Y H:i A', ], 'tables' => [ 'movements' => 'movements', 'holidays' => 'holidays', 'recurring_holidays' => 'recurring_holidays', ], 'models' => [ 'movement' => Movement::class, 'holiday' => Holiday::class, 'recurring_holiday' => RecurringHoliday::class, ], // Weekly work schedule — keyed by Day enum value (Sunday = 0 … Saturday = 6). // Days omitted are treated as non-working days. 'schedule' => [ Day::MONDAY->value => ['8:00', '17:00'], Day::TUESDAY->value => ['8:00', '17:00'], Day::WEDNESDAY->value => ['8:00', '17:00'], Day::THURSDAY->value => ['8:00', '17:00'], Day::FRIDAY->value => ['8:00', '15:30'], ], // Day-of-week values whose holidays substitute forward to the next working day // when observes_substitute = true on the row. Default empty = no substitution. // Malaysian state examples: // Sat-Sun states + post-2025 Johor → [Day::SUNDAY->value] // Kelantan, Terengganu → [Day::SATURDAY->value] // Kedah → [Day::FRIDAY->value] 'substitute' => [], // Status values to exclude from KPI calculation, keyed by movable morph type. // e.g. 'App\Models\Task' => ['except' => ['on_hold']]. 'status' => [ // 'App\Models\Task' => ['except' => ['on_hold']], ], ];
Calculating KPI
use Hasyirin\KPI\Facades\KPI; use Illuminate\Support\Carbon; $kpi = KPI::calculate( start: Carbon::parse('2025-01-01 08:00'), end: Carbon::parse('2025-01-03 15:30'), ); $kpi->minutes; // 990.0 — effective working minutes in range $kpi->hours; // 16.5 — minutes / 60 $kpi->period; // 2.0 — sum of (worked / scheduled) per day $kpi->metadata; // KPIMetadata — counts of scheduled / unscheduled / excluded days
Excluding dates
Holidays in the holidays table within the range are always excluded. You can also pass ad-hoc dates:
$kpi = KPI::calculate( start: Carbon::parse('2025-01-01 08:00'), end: Carbon::parse('2025-01-03 15:30'), excludeDates: [Carbon::parse('2025-01-02')], );
Overriding the schedule per call
use Hasyirin\KPI\Data\WorkSchedule; use Hasyirin\KPI\Enums\Day; $kpi = KPI::calculate( start: Carbon::parse('2025-01-06 09:00'), end: Carbon::parse('2025-01-06 15:00'), schedules: collect([ Day::MONDAY->value => WorkSchedule::parse(['9:00', '15:00']), ]), );
Holidays
The package recognizes two kinds of holiday rows, both contributing to exclusion in KPI::calculate().
One-off holidays
Holiday is a regular Eloquent model:
use Hasyirin\KPI\Models\Holiday; Holiday::create(['name' => 'New Year', 'date' => '2025-01-01']); Holiday::query()->range('2025-01-01', '2025-12-31')->get();
Recurring (fixed annual) holidays
For holidays that fall on the same gregorian month/day every year (Labour Day = May 1, Malaysian National Day = Aug 31), use RecurringHoliday:
use Hasyirin\KPI\Models\RecurringHoliday; RecurringHoliday::create([ 'name' => 'Labour Day', 'month' => 5, 'day' => 1, ]);
The calculator expands these to a concrete date for each year intersecting the calc range. Feb 29 is silently skipped in non-leap years.
Validity windows
effective_from and effective_until bound when the rule applies. Use them when a holiday was established or retired mid-stream — a 2015 calc should still exclude a holiday that ran 2010–2024, but a 2026 calc should not.
RecurringHoliday::create([ 'name' => 'Old Festival', 'month' => 6, 'day' => 15, 'effective_from' => '2010-01-01', 'effective_until' => '2024-12-31', ]);
Both bounds are nullable; null = unbounded.
Substitute day (next-working-day observance)
Both Holiday and RecurringHoliday carry an observes_substitute boolean. When true AND the row's day-of-week is listed in config('kpi.substitute') AND that day is non-working in the active schedule, the calculator observes the holiday on the next working day instead.
Substitute eligibility is configured at the package level because the same weekly schedule can have different substitute policies (Kelantan/Terengganu and Kedah both work Sun-Thu, but Kelantan substitutes a Saturday holiday while Kedah substitutes a Friday one):
// config/kpi.php // Most of Malaysia (Mon-Fri working, Sat-Sun off): 'substitute' => [Day::SUNDAY->value], // Kelantan, Terengganu (Sun-Thu working, Fri-Sat off): 'substitute' => [Day::SATURDAY->value], // Kedah (Sun-Thu working, Fri-Sat off): 'substitute' => [Day::FRIDAY->value],
RecurringHoliday::create([ 'name' => 'Labour Day', 'month' => 5, 'day' => 1, 'observes_substitute' => true, ]);
Multi-day skip and collisions
If the day immediately after the holiday is also non-working, the calculator keeps advancing until it finds a working day (Kedah's Fri → Sat (off) → Sun case is the canonical example).
The package does NOT chain substitutes through other holidays — if a substituted holiday lands on a day that is itself a holiday, both observe that same day (the calc loop dedups). If you want explicit chained observance (e.g., the federal Malaysian "next working day if Monday is also a public holiday" rule), add the chained day as its own one-off row.
Note on historical calculations
Movement::saving stores period and hours at completion time. Past completed movements are frozen — adding a new holiday row does NOT retroactively change them. This is intentional: a holiday introduced in 2026 should not be assumed to have existed in 2020.
Tracking movements on a model
A resource (any Eloquent model) can hold a forest of movements: multiple concurrent
open root tracks, each of which can have a tree of child movements. Receiver-based
dispatch decides whether pass() creates a root or a child.
Implement HasMovement and apply the trait to any model you want to track:
use Hasyirin\KPI\Concerns\InteractsWithMovement; use Hasyirin\KPI\Contracts\HasMovement; use Illuminate\Database\Eloquent\Model; class Task extends Model implements HasMovement { use InteractsWithMovement; }
$model->pass() — create a root
$task = Task::create([...]); $movement = $task->pass( status: 'open', sender: $system, // who triggered the transition (optional) actor: $user, // who is now responsible (optional) receivedAt: now(), // defaults to now() notes: 'Created via API', properties: ['source' => 'web'], supersede: null, // see "supersede" below expectsChildren: false, // see "expectsChildren" below );
BackedEnum statuses are accepted:
enum TaskStatus: string { case Open = 'open'; case InProgress = 'in_progress'; case Closed = 'closed'; } $task->pass(TaskStatus::InProgress, actor: $user);
$movement->pass() — create a child of the receiver
// $root is a Movement returned by $task->pass(...) $child = $root->pass('with_bob', actor: $bob); // $child has parent_id = $root->id, inherits movable_id/movable_type from $root. // $root is NOT auto-completed by this call.
To create a sibling of an existing child, call pass() on the parent:
$dave = $root->pass('with_dave', actor: $dave_user, supersede: false); // $dave is a child of $root, sibling of $child. // supersede: false keeps $child open alongside $dave.
To start a new concurrent root from a deeply nested context, go through movable:
$deep_child->movable->pass('audit', actor: $auditor);
Refreshing relations:
$movement->pass(...)does not auto-reload the receiver'schildrenrelation cache. If you need fresh state after appending children, call$root->load('children')or$root->refresh(). The trait's$model->pass()does reload$model->movementfor backwards compatibility with v1 single-chain callers.
$movement->complete() — close a movement
pass() no longer auto-completes prior movements by default (see supersede below).
Use complete() when you're done with a movement:
$movement->complete(); // closed at now() $movement->complete($at); // closed at a specific Carbon instance
If the movement has open descendants, they are cascade-closed with the same
completed_at timestamp. Each level computes its own period/hours via the
saving hook.
complete() is wrapped in a database transaction with lockForUpdate() on the
open-children query, making concurrent pass()/complete() interleaving safe.
Concurrency note:
pass()andcomplete()uselockForUpdate()to serialize operations on the same "previous" row, but a concurrent INSERT of a new sibling/root by a third transaction is not blocked. Two simultaneouspass()calls at the same level may both pick a "previous" that's about to be invalidated by the other, producing a staleprevious_idchain. If you require strict ordering, serialize at the application layer (queue worker, lock-based job, etc.).
supersede semantics
When you call $model->pass() or $movement->pass(), the same-level previous
movement (most recent open root for $model->pass(), most recent open sibling for
$movement->pass()) may be auto-closed. The supersede parameter controls this:
| Value | Behavior |
|---|---|
null (default) |
Close previous iff it has no open children and expects_children == false. |
true |
Always close previous — cascades through any open descendants. |
false |
Never close previous — the new movement runs concurrently. |
The default is convenient for sequential single-track workflows (a leaf root
naturally yields to its successor) while staying safe for trees (a parent with
open children, or a planned branch point with expects_children, is preserved).
expectsChildren
Marks a movement as a planned branch point. While expects_children = true, the
movement is protected from auto-supersession even when childless. The flag is
sticky — set once at creation, persists for the row's lifetime.
$agency = $file->pass('at_agency', actor: $agency_user, expectsChildren: true); // Later, even before $agency has any children: $file->pass('review', actor: $reviewer); // $agency stays open because expects_children = true.
passIfNotCurrent()
Only creates a new movement if the current one (most recent open root for
$model, most recent open child for $movement) doesn't already match the given
status and actor:
$movement = $task->passIfNotCurrent(TaskStatus::Open, actor: $user); // Movement instance on change, false on no-op.
Reading movements and the tree
$task->movement; // latest open movement of any depth (root or child), or null $task->movements; // MorphMany — full history, all depths $task->movements()->roots()->open()->get(); // open root tracks $task->movements()->roots()->closed()->get(); // historical roots
On a Movement:
$movement->parent; // BelongsTo Movement (null for roots) $movement->children; // HasMany Movement $movement->children()->open()->get(); $movement->children()->closed()->get(); $movement->previous; // BelongsTo Movement — same-level chain pointer $movement->movable; // MorphTo — back to the resource
Query scopes on Movement:
| Scope | Filter |
|---|---|
roots() |
whereNull('parent_id') |
open() |
whereNull('completed_at') |
closed() |
whereNotNull('completed_at') |
Computed attributes on Movement
period and hours are stored on save only when completed_at is set.
Incomplete movements have null for both; the on-the-fly accessors below
fall back to live calculation.
| Attribute | Description |
|---|---|
period |
Stored on save. Ratio of worked time to scheduled time on a completed movement. |
hours |
Stored on save. Worked time in hours. |
interval |
Accessor. hours * 3600 in seconds. |
formatted_period |
Accessor. Falls back to an on-the-fly calculation for incomplete movements. |
formatted_interval |
Accessor. Human-readable duration (e.g. 2 hours 15 minutes). |
formatted_received_at |
Accessor. received_at formatted via config('kpi.formats.datetime'). |
For trees: a parent's hours is inclusive — it covers the full duration the
parent was open, including time its open children were active. So
parent.hours ≥ Σ children.hours.
Events
Passed
Fires after every successful pass():
use Hasyirin\KPI\Events\Passed; Event::listen(function (Passed $event) { $event->current; // Movement that was just created $event->previous; // Movement that was just CLOSED by this pass via supersede, or null });
$previous is null when no supersession actually fired — i.e., when the
previous candidate had open children, was marked expects_children = true, or
when supersede: false was passed. Consumers wanting the chain pointer
regardless of closure should read $current->previous (the existing belongsTo).
Completed
Fires once per movement closure, regardless of trigger:
use Hasyirin\KPI\Events\Completed; Event::listen(function (Completed $event) { $event->movement; // the movement that just closed });
Triggers:
- Direct
$movement->complete(). - Cascaded close (parent's
complete()recursing through open children). - Supersession via
pass()(which internally callscomplete()on the prior).
Cascaded closures fire Completed for each descendant — useful for syncing per-movement state to external systems.
Testing
composer test
Changelog
Please see CHANGELOG for more information on what has changed recently.
Contributing
Please see CONTRIBUTING for details.
Security Vulnerabilities
Please review our security policy on how to report security vulnerabilities.
Credits
License
The MIT License (MIT). Please see License File for more information.