Cron Job In Laravel

Cron


Cron is a time-based task scheduler in Unix/Linux operating systems. It runs shell commands at a pre-specified time. Cron uses a configuration file called crontab also known as Cron table to manage the task scheduling process.

Crontab contains all the Cron jobs related to a specific task. Cron jobs are composed of two parts, the Cron expression, and a shell command that needs to be run.

* * * * * command/to/run


In the Cron expression above (* * * * *),each field is an option for determining the task schedule frequency. These options represent minute, hour, day of the month, month and day of the week in the given order. Asterisk symbol means all possible values. So, the above command will run every minute.

The Cron job below will be executed at 6:20 on the 10th of every month.

20 6 10 * * command/to/run


You can learn more about Cron job on Wikipedia. However, Laravel Cron Job Scheduling makes the whole process very easy.

Laravel Cron Job


Laravel Cron Job is an inbuilt task manager that gives your applications the ability to execute specific commands like sending a slack notification or removing inactive users at a periodic time. We will be using the latest version of Laravel, which is at the time of writing this article.

You need a Linux Operating System to run Cron Jobs. This tutorial also assumes a fair knowledge of PHP and Laravel.

Creating a new Project
In this tutorial, we will create a simple Laravel application to demonstrate task scheduling. Create a new Laravel project by running the following command.

composer create-project --prefer-dist laravel/laravel cron


Create your Scheduled Task in Laravel
There are different ways you can define scheduled tasks in Laravel. Let’s go through each of them to understand how they can be implemented in Laravel.

Create New Artisan Command
cd into your project and run the following command to create a new artisan command class:

php artisan make:command reallexiEmail


The above command will create a new command file, reallexiEmail.php, in the app/Console/Commands directory. Navigate to the file and you will find the following code in it:

 

namespace App\Console\Commands;

 

use Illuminate\Console\Command;

class reallexiEmail extends Command

{

/**

* The name and signature of the console command.

*

* @var string

*/

protected $signature ='command:name';

/**

* The console command description.

*

* @var string

*/

protected $description ='Command description';

/**

* Create a new command instance.

*

* @return void

*/

public function __construct()

{

parent::__construct();

}

/**

* Execute the console command.

*

* @return mixed

*/

public function handle()

{

//

}

}

In this code, the following line contains the name and the signature of the command.

Replace the words command:name with feed:import. This is what we will call this when running the command to perform the task.

protected $signature ='feed:import';


The above code also contains the description property. This is where you place the actual description of what this command will do. The description will be shown when the Artisan list command is executed alongside the signature. Change the description of the command to:

protected $description ='Send a Daily email to all users with a word and its meaning';

 

namespace App\Console\Commands;

use App\User;

use Illuminate\Console\Command;

use Illuminate\Support\Facades\Mail;

class reallexiEmail extends Command

{

/**

* The name and signature of the console command.

*

* @var string

*/

protected $signature ='feed:import';

/**

* The console command description.

*

* @var string

*/

protected $description ='Send a Daily email to all users with a word and its meaning';

/**

* Create a new command instance.

*

* @return void

*/

public function __construct()

{

parent::__construct();

}

/**

* Execute the console command.

*

* @return mixed

*/

public function handle()

{

$words = [

'aberration' => 'a state or condition markedly different from the norm',

'convivial' => 'occupied with or fond of the pleasures of good company',

'diaphanous' => 'so thin as to transmit light',

'elegy' => 'a mournful poem; a lament for the dead',

'ostensible' => 'appearing as such but not necessarily so'

];

// Finding a random word

$key = array_rand($words);

$value = $words[$key];

$users = User::all();

foreach ($users as $user) {

Mail::raw("{$key}-> {$value}", function ($mail) use ($user) {

$mail->from('[email protected]');

$mail->to($user->email)

->subject('Word of the Day');

});

}

$this->info('Word of the Day sent to All Users');

}

}

The above code will select a random word from the array and send emails to every user with the word. The handle method is called whenever the command is executed. This is where we place the code for doing the specific task. This is how the reallexiEmail.php file looks with the handle method and all other changes in place:

Registering the Command

Now that you have created the command, you will need to register it in the Kernel.

 

Go to app/Console/Kernel.php file that looks like this

 

namespace App\Console;

use Illuminate\Console\Scheduling\Schedule;

use Illuminate\Foundation\Console\Kernel as ConsoleKernel;

class Kernel extends ConsoleKernel

{

/**

* The Artisan commands provided by your application.

*

* @var array

*/

protected $commands = [

//

];

/**

* 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');

}

}

 

Change this file with the contents below. We have simply added our reallexiEmail class to the commands property and schedule it to run every day. In this file, we register the command class in the property and we schedule commands to be executed at periodic intervals in the method. This is where we handle all the Cron Jobs in Laravel.

 

namespace App\Console;

use Illuminate\Console\Scheduling\Schedule;

use Illuminate\Foundation\Console\Kernel as ConsoleKernel;

class Kernel extends ConsoleKernel

{

/**

* The Artisan commands provided by your application.

*

* @var array

*/

protected $commands = [

Commands\reallexiEmail::class,

];

/**

* Define the application's command schedule.

*

* @param \Illuminate\Console\Scheduling\Schedule $schedule

* @return void

*/

protected function schedule(Schedule $schedule)

{

$schedule->command('feed:import')

->daily();

}

/**

* Register the commands for the application.

*

* @return void

*/

protected function commands()

{

$this->load(__DIR__.'/Commands');

require base_path('routes/console.php');

}

}

Now, if you run the command in the terminal, you will see your command has been registered. You will be able to see the command name with the signature and description.

php artisan list

Setup, database and mail credentials in the file and make sure you have users in the database. Execute the command itself on the terminal:


php artisan feed:import

Command will be executed with the signature that is placed in protected $signature ='command:name'. You will also see the log message in the terminal.

Using Closure/Callback Method
Laravel Task Scheduler allows you to execute a callback or a closure periodically using the call method. Let’s add code in the schedule method of app/Console/Kernel.php Here’s the contents of the class with only the schedule method.

namespace App\Console;

use App\User;

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->call(function () {

User::where('spam_count', '>', 100)

->get()

->each

->delete();

})->hourly();

}

}

Exec Command We have passed the closure as the first parameter to the method. We have set the frequency of the task to hourly, so it will execute every hour. In the call the method, we are simply removing users with spam count of more than 100.

Laravel also allows you to schedule shell commands so that you could issue commands to the operating system and external applications. We use the method to execute a shell command.

Here’s a simple example, that allows you to take a backup of your code every month.

 

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(

'cp -r ' . base_path() . " " . base_path('../backups/' . date('jY'))

)->monthly();

}

}

Task Scheduler in Laravelexec the method will run the command you would pass as the first argument. In the code above, it simply copies your Laravel project to the backups folder. The folder will be followed by the day of the month and numeric representation of the year. You can also schedule Laravel Jobs in the Task Scheduler the same way we scheduled Artisan commands and closures. Let’s take a look at the Task Scheduler in detail.

Task Scheduler in Laravel executes the artisan command, shell, or a callback periodically on the defined time. To do this, we use the schedule method in app/Console/Kernel.php like we discussed earlier.

protected function schedule(Schedule $schedule)

{

$schedule->command('feed:import')

->daily();

}

 

 

Method    Description
->cron(‘* * * * * *’);    Run the task on a custom Cron schedule
->everyMinute();    Run the task every minute
->everyFiveMinutes();    Run the task every five minutes
->everyTenMinutes();    Run the task every ten minutes
->everyFifteenMinutes();    Run the task every fifteen minutes
->everyThirtyMinutes();    Run the task every thirty minutes
->hourly();    Run the task every hour
->hourlyAt(17);    Run the task every hour at 17 mins past the hour
->daily();    Run the task every day at midnight
->dailyAt(’13:00′);    Run the task every day at 13:00
->twiceDaily(1, 13);    Run the task daily at 1:00 & 13:00
->weekly();    Run the task every week
->weeklyOn(1, ‘8:00’);    Run the task every week on Tuesday at 8:00
->monthly();    Run the task every month
->monthlyOn(4, ’15:00′);    Run the task every month on the 4th at 15:00
->quarterly();    Run the task every quarter
->yearly();    Run the task every year
->timezone(‘America/New_York’);    Set the timezone


$schedule->command('feed:import') is where we define which command needs to be executed and "->daily();" defines the frequency of execution. There are some more time intervals that we can define. You can replace " ->daily();" with another time interval option from the following list. You can find more about Task Scheduling in Laravel Documentation.

Starting the Laravel Scheduler
Let’s set up the Cron Jobs to run automatically without initiating manually by running the command. To start the Laravel Scheduler itself, we only need to add one Cron job which executes every minute. Go to your terminal, ssh into your server, cd into your project and run this command.


crontab -e

This will open the server Crontab file, paste the code below into the file, save and then exit.

* * * * * cd ~/path-to-your-project && php artisan schedule:run >> /dev/null 2>&1

 

By: Mutasem Elayyoub