🔭 Want to get alerted when your Laravel scheduled tasks stop working?
Head over to Better Uptime start monitoring your jobs in 2 minutes
Posted on: February 08, 2024 12:26 PM
Posted by: Renato
Categories: Laravel PHP jobs scheduling monitoring
Views: 336
O agendamento de tarefas é uma técnica útil para automatizar várias tarefas repetitivas com base em um agendamento. Essas tarefas ou trabalhos podem ser de natureza crítica (como fazer backup de um banco de dados) ou podem ser tão simples quanto enviar um e-mail semanal para você ou seus clientes. A execução manual desses trabalhos pode se tornar entediante muito rapidamente, portanto, a melhor solução é automatizá-los e monitorá-los para que possam ser executados de maneira previsível e oportuna.
A maneira tradicional de agendar tarefas em aplicativos implantados em servidores Linux é por meio do utilitário cron. Contudo, tal sistema deve ser implementado separadamente da aplicação, o que pode ser bastante limitante.
Laravel oferece uma abordagem holística para agendamento de tarefas através de seu agendador de comandos que permite agendar tarefas dentro do próprio aplicativo. Neste tutorial, exploraremos como criar jobs agendados no Laravel e também implementaremos uma solução de monitoramento para ajudá-lo a notificá-lo imediatamente se uma tarefa agendada falhar ou não for executada conforme o esperado.
Task scheduling is a useful technique for automating various repetitive tasks based on a schedule. Such tasks or jobs may be mission critical in nature (such as backing up a database), or it may be as simple as send a weekly email to yourself or your customers. Manually running such jobs can get tedious real quick so a better solution is to automate and monitor them so they can run predictably in a timely fashion.
The traditional way to schedule tasks on applications deployed to Linux servers is through the cron utility. However, such system must be implemented separately from the application which can be quite limiting.
Laravel offers a holistic approach to task scheduling through its command scheduler which allows you to schedule tasks within the application itself. In this tutorial, we will explore how to create scheduled jobs in Laravel, and we will also implement a monitoring solution to help you to promptly notify you if a scheduled task fails or doesn't run as expected.
Head over to Better Uptime start monitoring your jobs in 2 minutes
Before proceeding with this article, ensure that you have access to a Linux machine with recent versions of Cron, PHP , and Composer installed.
You should also create a new Laravel project so that you may test the code snippets in this tutorial. Assuming PHP and Composer are installed, you can proceed to create a Laravel project using the following command:
composer create-project laravel/laravel <project_name>
Your new Laravel project should have the following structure:
.
├── README.md
├── app
│ ├── Console
│ │ └── Kernel.php
│ ├── Exceptions │ │ └── Handler.php │ ├── Http │ │ ├── Controllers │ │ │ ├── Controller.php │ │ ├── Kernel.php │ │ └── Middleware │ ├── Models │ └── Providers ├── artisan ├── bootstrap ├── composer.json ├── composer.lock ├── config ├── database ├── package.json ├── php_errors.log ├── phpunit.xml ├── public ├── resources ├── routes ├── storage ├── tests └── vite.config.js
For this tutorial, we only care about the Console
directory because all scheduled jobs in Laravel are defined in the app\Console\Kernel.php
file.
Go ahead and open the file with the following command:
nano app\Console\Kernel.php
<?php
namespace App\Console;
use Illuminate\Console\Scheduling\Schedule;
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
class Kernel extends ConsoleKernel
{
/**
* Define the application's command schedule.
*
* @param \Illuminate\Console\Scheduling\Schedule $schedule
* @return void
*/
protected function schedule(Schedule $schedule)
{
// $schedule->command("inspire")->hourly();
}
/**
* Register the commands for the application.
*
* @return void
*/
protected function commands()
{
$this->load(__DIR__ . "/Commands");
require base_path("routes/console.php");
}
}
Notice the highlighted schedule()
function above. It's where you can schedule your tasks, and Laravel provides an example of how it is done within the function:
$schedule->command("inspire")->hourly();
This line is commented out by default, but you should uncomment it before going forward. The $schedule
variable is an instance of the Schedule
class built into Laravel. Its command()
method will schedule an Artisan command which displays an inspiring quote in the console.
To view a list of all available Artisan commands, run the command below in the terminal from your project root:
php artisan list
You will observe the command list alongside a brief description of what they do:
Laravel Framework 9.33.0
. . .
Available commands:
about Display basic information about your application
clear-compiled Remove the compiled class file
. . .
view
view:cache Compile all of the application's Blade templates
view:clear Clear all compiled view files
After choosing a command, you must specify how often you wish to execute it. In the default example, the hourly()
method executes the inspire
command every hour. For demonstration purposes, change it to everyMinute()
so you can see the effect of the task scheduling much quicker.
$schedule->command('inspire')->everyMinute();
After making the change, invoke the scheduler by executing the command below:
php artisan schedule:run
The schedule:run
command will go through all scheduled tasks and determine if Laravel should execute the task based on the current time. For example, hourly()
will schedule a task to be executed on the hour mark (xx:00) while everyFiveMinutes()
executes the job when the current time is xx:x0, xx:x5, xx:10, etc. If no task is currently scheduled to be executed, the following output will be printed to the terminal:
INFO No scheduled commands are ready to run.
However, since we are using everyMinute()
in our example, you should observe that the task is executed and the following output is printed to the terminal:
2022-10-06 21:42:39 Running ['artisan' inspire] ...................2,878ms DONE
⇂ '/usr/bin/php8.1' 'artisan' inspire > '/dev/null' 2>&1
Notice that a log message describing the executed command is printed, but the result isn't displayed. That's because Laravel automatically ignores the output of a scheduled task by forwarding it to /dev/null
(see second line of above output) since they are typically run in the background.
If you need to utilize the output of your scheduled task, you can place it in a file for later inspection through the sendOutputTo()
method:
$schedule
->command("inspire")
->everyMinute()
->sendOutputTo("scheduler-output.log");
After making the above change, invoke the scheduler again and view the contents of the scheduler-output.log
file that is subsequently created:
cat scheduler-output.log
“ Knowing is not enough; we must apply. Being willing is not enough; we must do. ”
— Leonardo da Vinci
Note that the sendOutputTo()
method overwrites the contents of its file argument for each invocation. If this is not desired, you can use the appendOutputTo()
method that appends the command output to the end of the file instead.
$schedule
->command("inspire")
->everyMinute()
->appendOutputTo("scheduler-output.log");
Run the scheduler a few times, and observe that the file is no longer being overwritten:
cat scheduler-output.log
“ Simplicity is the consequence of refined emotions. ”
— Jean D'Alembert
“ Order your soul. Reduce your wants. ”
— Augustine
“ Simplicity is the essence of happiness. ”
— Cedric Bledsoe
“ I begin to speak only when I am certain what I will say is not better left unsaid. ”
— Cato the Younger
A problem with the schedule:run
command is that it only invokes the scheduler once which means that the scheduled tasks are executed just once. To ensure tasks are run as scheduled, the scheduler needs to be constantly running and this is achieved by using the schedule:work
command instead as shown below:
php artisan schedule:work
This command will run in the foreground and invoke the schedule:run
command every minute until it is terminated by pressing Ctrl-C
. It is useful for testing your code during development so you don't have to manually invoke the scheduler each time.
In a production environment, you should execute the scheduler in the background through a Cron Job as shown below:
crontab -e
# Edit this file to introduce tasks to be run by cron.
. . .
* * * * * cd <path_to_your_project> && php artisan schedule:run >> /dev/null 2>&1
Aside from Artisan commands, you can also schedule the execution of any system command using the exec()
method as follows:
$schedule->exec("bash script.sh")->hourly();
The sendOutputTo()
and appendOutputTo()
methods discussed earlier can also be used to capture the output of system commands.
If you need to schedule a function in Laravel, you can use the call()
method and pass in a closure that wraps the function as follows:
<?php
namespace App\Console;
use Illuminate\Console\Scheduling\Schedule;
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
use Illuminate\Support\Facades\DB;
class Kernel extends ConsoleKernel
{
/**
* Define the application's command schedule.
*
* @param \Illuminate\Console\Scheduling\Schedule $schedule
* @return void
*/
protected function schedule(Schedule $schedule)
{
$schedule
->call(function () {
DB::table("users")
->where("status", "inactive")
->delete();
})
->daily();
}
}
The scheduled job will run daily in this example and delete all inactive users.
In addition to scheduling closures, you may also schedule invokable objects with the call()
method. These are PHP classes that contain an __invoke()
method. When a new instance of the class is fed to the call()
method, its __invoke()
method will be executed according to the specified schedule.
$schedule->call(new SomeClass)->hourly();
Queued jobs are typically long-running tasks that are processed in the background to avoid interfering with the application's main processes. Such tasks can be executed on a defined schedule by using the job()
method on the scheduler.
For instance, you can use this feature to update the search index of your application once every hour as follows:
use App\Jobs\UpdateSearchIndex;
$schedule->job(new UpdateSearchIndex)->hourly();
If you are using a third-party service (such as Amazon SQS) to handle your queued jobs, you also need to specify the name of the job and the service you are using so that the appropriate queue connection will be used to queue the job:
$schedule->job(new SomeQueuedJob, "myJob", "sqs")->hourly();
You can obtain a list of all scheduled tasks using the command below. It displays each scheduled task and the time of their next invocation:
php artisan schedule:list
0 * * * * php artisan inspire ............................. Next Due: 6 minutes from now
0 * * * * bash script.sh ...................................Next Due: 6 minutes from now
0 0 * * * Closure at: app/Console/Kernel.php:23 ............Next Due: 4 hours from now
0 * * * * App\Jobs\UpdateSearchIndex .......................Next Due: 6 minutes from now
When you have multiple tasks scheduled to run simultaneously, Laravel will execute them one by one according to how they are defined in the app/Console/Kernel.php
file. However, if one task requires a long time to run, it will delay all subsequent tasks. Therefore, you might want to run your scheduled tasks in the background using the runInBackground()
method, so that multiple jobs can run simultaneously.
$schedule
->exec("bash script.sh")
->hourly()
->runInBackground();
Note that runInBackground()
can only be used for tasks scheduled with the exec()
or command()
methods only.
So far, we've only seen the hourly()
and everyMinute()
methods in action, but Laravel offers several other frequency options, which can be broadly classified into a few different categories which are discussed below.
$schedule->command("inspire")->everyMinute(); // Execute every minute
$schedule->command("inspire")->everyTwoMinutes(); // Execute every two minutes
$schedule->command("inspire")->everyThreeMinutes();
$schedule->command("inspire")->everyFourMinutes();
$schedule->command("inspire")->everyFiveMinutes();
$schedule->command("inspire")->everyTenMinutes();
$schedule->command("inspire")->everyFifteenMinutes();
$schedule->command("inspire")->everyThirtyMinutes();
$schedule->command("inspire")->hourly(); // Execute every hour
$schedule->command("inspire")->hourlyAt(15); // Execute every hour at minute 15
$schedule->command("inspire")->everyOddHour();
$schedule->command("inspire")->everyTwoHours();
$schedule->command("inspire")->everyThreeHours();
$schedule->command("inspire")->everyFourHours();
$schedule->command("inspire")->everySixHours();
The hour-based options above will run their tasks at the first minute of the hour, but the hourlyAt(n)
method can be used to run a task at the nth
minute of the hour instead.
$schedule->command("inspire")->daily(); // Execute daily at 00:00
$schedule->command("inspire")->dailyAt("10:15"); // Execute daily at 10:15
$schedule->command("inspire")->twiceDaily(10, 16); // Execute twice daily at 10:00 and 16:00
$schedule->command("inspire")->twiceDailyAt(10, 16, 15); // Execute twice daily at 10:15 and 16:15
$schedule->command("inspire")->weekly(); // Execute weekly on Sunday at 00:00
$schedule->command("inspire")->weeklyOn(2, "8:00"); // Execute weekly on Tuesday at 08:00
The weeklyOn()
method allows you to specify a day (Sunday:0
, Monday:1
, ... , Saturday:6
) and a time, but you can also specify multiple days using an array:
$schedule->command("inspire")->weeklyOn([2, 4, 5], "8:00"); // Execute weekly on Tuesday, Thursday and Friday at 08:00
$schedule->command("inspire")->monthly(); // Execute on the first day of every month at 00:00
$schedule->command("inspire")->monthlyOn(4, "15:00"); // Execute on the fourth day of every month at 15:00
$schedule->command("inspire")->twiceMonthly(1, 16, "13:00"); // Execute on the first and sixteenth day of every month at 13:00
$schedule->command("inspire")->lastDayOfMonth("15:00"); // Execute on the last day of every month at 15:00
$schedule->command("inspire")->quarterly(); // Execute on the first day of every quarter at 00:00
$schedule->command("inspire")->yearly(); // Execute on the first day of every year at 00:00
$schedule->command("inspire")->yearlyOn(6, 1, "17:00"); // Execute on June 1st of every year at 17:00
Constraint options are a special set of methods that defines additional constraints after you have specified the frequency. For example, you can first schedule a job to run weekly and then set constraints like this:
$schedule->command("inspire")->weekly()->weekdays(); // Runs every Monday to Friday at 00:00
$schedule->command("inspire")->weekly()->weekends(); // Limit the task to weekends alone
$schedule->command("inspire")->weekly()->sundays();
$schedule->command("inspire")->weekly()->mondays();
$schedule->command("inspire")->weekly()->tuesdays();
$schedule->command("inspire")->weekly()->wednesdays();
$schedule->command("inspire")->weekly()->thursdays();
$schedule->command("inspire")->weekly()->fridays();
$schedule->command("inspire")->weekly()->saturdays();
$schedule->command("inspire")->weekly()->days([2, 5]); // Execute every Tuesday and Friday at 00:00
The days()
method takes an array of integers as input, which allows you to limit the task execution to specific days of the week.
Besides the day constraints, there are two additional time-based constraint methods, between()
and unlessBetween()
, which defines a range of times that a job is allowed to be executed.
// Execute hourly on weekdays, between 08:00 and 17:00
$schedule
->command("inspire")
->weekdays()
->hourly()
->between("8:00", "17:00");
// Execute hourly on weekdays, before 08:00 and after 17:00
$schedule
->command("inspire")
->weekdays()
->hourly()
->unlessBetween("8:00", "17:00");
A major benefit of using Laravel to schedule tasks is that you can use something other than the current time to trigger the execution of the tasks. For example, the when()
method takes a callback function as its input, and the task will only run if the function returns true
:
$schedule
->command("inspire")
->hourly()
->when(function () {
$weather = . . .;
if ($weather == "sunny") {
return true;
} else {
return false;
}
});
The above example schedules a task for execution based on the weather. It only executes on sunny days. The one below uses the skip()
method to defines condition for skipping a scheduled task.
$schedule
->command("inspire")
->hourly()
->skip(function () {
$weather = . . .;
if ($weather == "rainy") {
return true;
} else {
return false;
}
});
Lastly, you may also schedule tasks based on the current application environment, which is helpful if you want to schedule some tasks only in production.
$schedule
->command("inspire")
->weekdays()
->environments(["staging", "production"]);
In the above example, the task will only run in staging and production environments.
The scheduling methods discussed so far should be enough for most scenarios, but Laravel offers some advanced techniques for creating more complex schedules.
cron()
methodThe cron()
method allows you to create schedules using raw Cron expressions. For example:
$schedule->command("inspire")->cron("5 4 * * sun");
This code will schedule the inspire
Artisan command to run at 04:05 every Sunday.
at()
methodYou may have noticed that some of the frequency methods (such as mondays()
, quarterly()
, and others) do not allow you to specify an exact time, which can sometimes be limiting. This can be fixed through the at()
method as shown below:
$schedule
->command("inspire")
->weekly()
->wednesdays()
->at("13:15"); // Executes every Wednesday at 13:15
timezone()
methodIf you'd like to schedule a task in a different timezone, you can also chain a timezone()
method in your schedule like this:
$schedule
->command("inspire")
->timezone("America/New_York")
->at("2:00")
Another useful feature that Laravel provides for task scheduling is task hooks. They let you execute some code before or after the task is executed, or if a certain condition is true (such as if the task failed or succeeded).
For example, you can log a message before and after a task is executed as shown below. Please refer to the linked article for more information on logging in Laravel. We assume you've already configured the logging system correctly for demonstration purposes.
use Illuminate\Support\Stringable;
$schedule
->exec("bash scripts/backup.bash")
->before(function () use ($logger) {
Log::info("The database backup script executed at " . time());
})
->after(function (Stringable $output) {
Log::info($output);
})
->everyMinute();
If your scheduled job produces some output, you can access it in the after()
method through the $output
variable with the Illuminate\Support\Stringable
type.
A scheduled tasks might succeed or fail, so Laravel also provides the onSuccess()
and onFailure()
hooks for dealing with either outcome:
$schedule
->command(". . .")
->before(function () use ($logger) {
Log::info("The script weather.php executed at " . time());
})
->onSuccess(function (Stringable $output) {
// The task succeeded...
})
->onFailure(function (Stringable $output) {
// The task failed...
})
->everyMinute();
Just like the after()
hook, you can access the output through Stringable $output
. Note that the onSuccess()
and onFailure()
hooks only work for tasks scheduled with the command()
or exec()
methods because task failure is detected through a non-zero exit code.
Finally, we have the ping
family of hooks which are useful for notifying an external service when a scheduled task has began, completed, or failed. Here are the available methods and their signatures:
pingBefore($url)
thenPing($url)
pingBeforeIf($condition, $url)
thenPingIf($condition, $url)
pingOnSuccess($url)
pingOnFailure($url)
Now that we've covered several Laravel task scheduling essentials, let's look at some practical examples of how it can be useful in a real-life project. We'll automate some tasks for a class management app where teachers can record the student's grades for each class.
Start by cloning the project to your computer using the following command:
git clone https://github.com/betterstack-community/class-management-app.git
Next, change into the class-management-app
directory:
cd class-management-app
Rename the .env.example
file at the project root to .env
:
mv .env.example .env
Afterward, install the required dependencies with composer
:
composer install
Generate a new APP_KEY
by running the following Artisan command:
php artisan key:generate
Before starting the development server, ensure all the required modules are enabled in your php.ini
file:
[php.ini]
extension=pdo.so
extension=pdo_sqlite.so
extension=sqlite.so
. . .
The location of your php.ini file can be found by running:
php -i | grep "Loaded Configuration File"
Finally, start the dev server:
php artisan serve
INFO Server running on [http://127.0.0.1:8000].
Press Ctrl+C to stop the server
Open your browser and go to http://127.0.0.1:8000 . You should see the home page of the class management app. To save time, we've included some dummy data for the project.
You can then create new students:
Or update/delete students:
Let's begin by scheduling a common task that is required in almost all production web applications: a daily database backup. Since the SQLite database is being utilized for this project, backing it up only involves copying the database.sqlite
file to a specified location.
You can write a shell script to perform this task as shown below:
mkdir scripts && nano scripts/backup.sh
#!/usr/bin/env bash
# Get current date in yyyy-mm-dd format
NOW=$(date "+%Y-%m-%d")
# If backup directory does not exist, create one
if [ ! -d "backup" ]; then
mkdir "backup"
fi
# Copy database.sqlite into backup directory
cp "database.sqlite" "backup/${NOW}-database.sqlite"
# Find and delete files older than 7 days
find "backup" -type f -mtime +7 -delete
When this script is executed, it will backup the database by copying the database.sqlite
into the backup
directory under a new name containing the current date. It also scans for files older than seven days and deletes them to prevent the backups directory from growing too large. This isn't much of a backup, but it suffices to demonstrate the concept of automating such a process.
Go ahead and schedule this script to run once a day with the following code:
<?php
namespace App\Console;
use Illuminate\Console\Scheduling\Schedule;
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
class Kernel extends ConsoleKernel
{
/**
* Define the application's command schedule.
*
* @param \Illuminate\Console\Scheduling\Schedule $schedule
* @return void
*/
protected function schedule(Schedule $schedule)
{
$schedule->exec("bash scripts/backup.sh")->daily();
}
/**
* Register the commands for the application.
*
* @return void
*/
protected function commands()
{
$this->load(__DIR__ . "/Commands");
require base_path("routes/console.php");
}
}
You can run the schedule:list
command to see the next time it is scheduled to run:
php artisan schedule:list
0 0 * * * bash scripts/backup.sh .............................. Next Due: 12 hours from now
If you want to confirm that the script is working properly, you can change the frequency to from daily()
to everyMinute()
and run the schedule:work
command so that the scheduler keeps running every minute:
After a minute or so, you should observe the generated backup
directory as well as the backup files:
A classic use case for task scheduling is automating the generation of reports for relevant business metrics. In this section, we will demonstrate how to automate such tasks by sending a weekly report of students' performance to an email address. In this example, the report will be in the same format as the application's homepage.
Before proceeding, you must configure Laravel's emailing functionality with Gmail SMTP (or any email provider of your choosing). Start by heading to Google My Account → Security , and enabling 2-Step Verification. Afterward, go to App passwords and create a unique password for your Laravel application. Under Select app, choose the Other (Custom name) option and type "Class Management App" in the text input.
Once the password is generated, copy it to your clipboard. Head back to your text editor and open your .env
file once again. Edit the MAIL_
section of the file as shown below. Note that the MAIL_PASSWORD
must be the unique password you just created.
. . .
MAIL_DRIVER=smtp
MAIL_HOST=smtp.googlemail.com
MAIL_PORT=465
MAIL_USERNAME=<your_gmail_address>
MAIL_PASSWORD=<your_password>
MAIL_ENCRYPTION=ssl
MAIL_FROM_ADDRESS="<your_gmail_address>"
MAIL_FROM_NAME="${APP_NAME}"
. . .
Next, head to your app/Console/Kernel.php
file, and modify it as shown below. Don't forget to replace the <your_email>
placeholder with an actual email address.
<?php
namespace App\Console;
use Illuminate\Console\Scheduling\Schedule;
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
use Illuminate\Support\Facades\Mail;
use App\Mail\StudentFailClass;
use App\Mail\WeeklyReport;
use App\Models\Student;
class Kernel extends ConsoleKernel
{
/**
* Define the application's command schedule.
*
* @param \Illuminate\Console\Scheduling\Schedule $schedule
* @return void
*/
protected function schedule(Schedule $schedule)
{
. . .
$schedule
->call(function () {
Mail::to("<your_email>")->send(
new WeeklyReport(Student::all())
);
})
->weekly();
}
. . .
}
The WeeklyReport
class has not been created yet, so create it by pasting the code below in a new app/Mail/WeeklyReport.php
file:
<?php
namespace App\Mail;
use App\Models\Student;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Mail\Mailable;
use Illuminate\Mail\Mailables\Content;
use Illuminate\Mail\Mailables\Envelope;
use Illuminate\Queue\SerializesModels;
class WeeklyReport extends Mailable
{
use Queueable, SerializesModels;
/**
* The student instance.
*
* @var \App\Models\Student
*/
public $students;
/**
* Create a new message instance.
*
* @return void
*/
public function __construct($students)
{
$this->students = $students;
}
/**
* Get the message envelope.
*
* @return \Illuminate\Mail\Mailables\Envelope
*/
public function envelope()
{
return new Envelope(subject: "Weekly Report");
}
/**
* Get the message content definition.
*
* @return \Illuminate\Mail\Mailables\Content
*/
public function content()
{
return new Content(view: "index");
}
/**
* Get the attachments for the message.
*
* @return array
*/
public function attachments()
{
return [];
}
}
When the call()
method is executed, Laravel will send the index
page (resources/views/index.blade.php
) to the email of your choice, displaying everyone in the classes and their grade for each subject.
You can temporarily change the schedule frequency to something more suitable for testing such as everyMinute()
for example, and wait for the task to execute. You should receive the following email in your inbox:
- Fonte: https://betterstack.com/community/guides/scaling-php/laravel-task-scheduling/
Donate to Site
Developer