codeldev / laravel-job-log
A simple Laravel package that automatically logs all jobs queued and processed by your application to the database.
Fund package maintenance!
Requires
- php: ^8.4
- illuminate/contracts: *
- spatie/laravel-package-tools: ^1.93
Requires (Dev)
- larastan/larastan: ^3.9.5
- laravel/pint: ^1.29
- nunomaduro/collision: ^8.9.3
- nunomaduro/pao: 0.x-dev
- orchestra/testbench: ^11.1.0
- peckphp/peck: ^0.3.0
- pestphp/pest: ^4.5.0
- pestphp/pest-plugin-arch: ^4.0.2
- pestphp/pest-plugin-laravel: ^4.1
- pestphp/pest-plugin-type-coverage: ^4.0.4
- phpstan/extension-installer: ^1.4.3
- phpstan/phpstan-deprecation-rules: ^2.0.4
- phpstan/phpstan-phpunit: ^2.0.16
- rector/rector: ^2.4.1
- roave/security-advisories: dev-latest
This package is auto-updated.
Last update: 2026-04-20 04:00:31 UTC
README
A simple Laravel package that automatically logs all jobs queued and processed by your application to the database. It listens to Laravel's built-in Job Events to record information without wrapping or modifying the Queue itself. Ships with configurable table names, a swappable Eloquent model, and a built-in prune command to manage retention.
Requirements
- PHP 8.4+
- Laravel 13+
Installation
You can install the package via composer:
composer require codeldev/laravel-job-log
You can publish and run the migrations with:
php artisan vendor:publish --tag="job-log-migrations"
php artisan migrate
You can publish the config file with:
php artisan vendor:publish --tag="job-log-config"
This is the contents of the published config file:
return [ 'prune_days' => (int) env('JOB_LOG_PRUNE_DAYS', 365), 'model' => CodelDev\LaravelJobLog\Models\LaravelJobLog::class, 'table' => env('JOB_LOG_TABLE', 'job_log'), ];
Environment Variables
The following env variables are available to configure the package using your env file.
JOB_LOG_PRUNE_DAYS=365 JOB_LOG_TABLE=job_log
Usage
Once installed, the package automatically logs every job. No additional setup is required.
Pruning Old Records
Add to your routes/console.php file:
Schedule::command('job-log:prune') ->weeklyOn(1, '02:30') ->withoutOverlapping();
Run manually:
php artisan job-log:prune
Querying the Data
The package provides an Eloquent model you can use directly:
use CodelDev\LaravelJobLog\Models\LaravelJobLog; use CodelDev\LaravelJobLog\Enums\LaravelJobRunStatusEnum; // Get all logs LaravelJobLog::all(); // Filter by status LaravelJobLog::query() ->where('status', LaravelJobRunStatusEnum::SUCCEEDED) ->get(); LaravelJobLog::query() ->where('status', LaravelJobRunStatusEnum::FAILED) ->get(); LaravelJobLog::query() ->where('status', LaravelJobRunStatusEnum::RUNNING) ->get(); // Filter by job class LaravelJobLog::query() ->where('job', 'App\\Jobs\\MyJob') ->get(); // Filter by queue LaravelJobLog::query() ->where('queue', 'default') ->get(); // Jobs from the last 24 hours LaravelJobLog::query() ->where('started_at', '>=', now()->subDay()) ->get(); // Slow jobs (over 1 second) LaravelJobLog::query() ->where('duration_ms', '>', 1000) ->orderBy('duration_ms', 'desc') ->get(); // Failed jobs with exception details LaravelJobLog::query() ->where('status', LaravelJobRunStatusEnum::FAILED) ->with('failedJob') ->get(); // Paginate results LaravelJobLog::query() ->orderBy('created_at', 'desc') ->paginate(15);
Available Fields
LaravelJobLog
| Field | Type | Description |
|---|---|---|
id |
string |
UUID primary key |
job_uuid |
string|null |
Laravel's internal job UUID |
job |
string |
Fully qualified job class name |
queue |
string |
Queue name (e.g. default) |
attempt |
int |
Attempt number |
started_at |
CarbonImmutable |
When the job started processing |
completed_at |
CarbonImmutable|null |
When the job finished |
status |
LaravelJobRunStatusEnum |
RUNNING (1), SUCCEEDED (2), or FAILED (3) |
duration_ms |
int|null |
Execution time in milliseconds |
failed_job_id |
int|null |
Foreign key to Laravel's failed_jobs table |
created_at |
CarbonImmutable|null |
Record creation timestamp |
updated_at |
CarbonImmutable|null |
Record update timestamp |
LaravelJobFailed (read-only view of Laravel's failed_jobs table)
| Field | Type | Description |
|---|---|---|
id |
int |
Auto-incrementing primary key |
uuid |
string |
Job UUID |
connection |
string |
Queue connection name |
queue |
string |
Queue name |
payload |
string |
Serialized job payload |
exception |
string |
Exception message and stack trace |
failed_at |
CarbonImmutable |
When the job failed |
The LaravelJobLog model has a failedJob() relationship that links to LaravelJobFailed via the failed_job_id column, giving you access to the full exception details for failed jobs.
Using a Custom Model
You can extend the package model to add your own behaviour, scopes, or relationships. Create your custom model, extend the package model, then update the config:
use CodelDev\LaravelJobLog\Models\LaravelJobLog as BaseJobLog; class JobLog extends BaseJobLog { public function scopeFailed($query) { return $query->where('status', LaravelJobRunStatusEnum::FAILED); } }
Then in config/job-log.php:
'model' => \App\Models\JobLog::class,
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.