Cron Jobs with systemd.timer

Published: November 15, 2014 · Previous · Next

https://wiki.archlinux.org/index.php/Systemd/Timers

Timers are systemd unit files whose name ends in .timer that control .service files or events. Timers have the ability to be an alternate to cron. Timers have built-in support for calendar time events, monotonic time events, and have the ability to run asynchronously.

There are lots of flamewars against systemd; mostly introduced by people who grew up with e. g. sysvinit scripts. Besides these flamewars systemd provides a very well documented and very easy to use out of the box solution for many tasks of an administrator.

At first we have to clarify a few terms and definitions. Systemd grows from release to release and it has a lot of different concepts we have to deal with. The first concept I have to introduce for a cron replacement with systemd is systemd.service.

systemd.service

Service files are commonly used to start and stop programs. The systemctl tool provides the necessary commands to control your services, such as start, stop, restart, reload, and so on. It is your job (or the job of the package maintainer) to write an appropriate service file so that systemctl is able to map e.g. the start command to the corresponding shell command.

Let's do a simple example. Somebody has written an awesome backupscript and he wants that program to be executed daily. The start command for that piece of software might be something like:

$ awesome_program start

So we have to tell systemd that command first. Don't forget to put the full path of your script into ExecStart. Otherwise systemd is not able to find it!

/etc/sytemd/system/awesome.service

[Unit]
Description=My awesome backup script
After=network.target

[Service]
Type=oneshot
ExecStart=/path/to/awesome_programm start

We call this file awesome.service. The structure of systemd service files is self-explanatory. The [Unit] section is used to define some metadata such as the description or dependencies. The [Service] section actually defines the start command of our service. As we are most likely using a script which runs and exists after finishing its tasks we have to set the Type to oneshot, see also systemd.service(5). Maybe I will give more information about that topic in another articles.

Now there is the question where we are supposed to put this file. This topic is covered in detail in systemd.unit(5). At the moment we are just interested in the local configuration path when running in system mode: /etc/systemd/system. So let's copy this service to it's intended location /etc/systemd/system/awesome.service.

systemd.timer

In the previous section we have covered the required steps to design a simple systemd service which is used to start our script. Now we will add the unit which is used to trigger the service. The timer has to be named according to the corresponding service awesome.timer; we put this file in the same directory.

/etc/sytemd/system/awesome.timer

[Unit]
Description=Weekly trigger for my awesome program

[Timer]
OnCalendar=daily
Persistent=true

[Install]
WantedBy=multi-user.target

The timer file is self-explanatory as well. It has three required sections: [Unit], [Timer] and [Install].

I will now focus on the [Timer] section. It is used to define when the timer should be activated. At first there is the OnCalender option. It understands a few keywords such as minutely, hourly, daily, monthly, weekly, yearly, quarterly, semiannually. If these keywords fit into your requirements everything is fine; more fine graned control is provided by systemd.time(7). Maybe you want to run your service every six hours. So you have to define the OnCalendar option like this:

OnCalendar=*-*-* 0/6:42:42

This means in detail (according to systemd.time(7)):

Run on

The repeating is defined with the slash. An OnCalender which runs every six minutes instead would look like this:

OnCalendar=*-*-* 0:0/6:42

Just for the sake of completeness, Persistent=true means that any missed timer will run after bootup. It is useful to catch up on missed runs of the service when the machine was off. It is compareable to anacron.

Last but not least there is the Install section left. This one is needed for the most important part: enabling the timer. We do this with systemctl enable awesome.timer. Installing the timer into the multi-user.target just means we want it to be started during bootup. If you want to dive into more detail check out the Arch Wiki. As we do not want to reboot now we also have to start the timer using

$ systemctl start awesome.timer

Listing timers

Let's check if everything works as expected. Systemctl has a built in command to list and inspect timers on your system.

$ systemctl list-timers

will tell you which timers have been recognized by systemd.

Where is the user crontab?

Maybe you have wondered why you have to copy every file into system directories. As mentioned before this is for systemd running in system mode. These services and timers are started and enabled system wide for all users on the machine. If you do not have sudo privileges or you just want to define your own timer you can use systemd's user mode.

https://wiki.archlinux.org/index.php/Systemd/Timers

systemd offers users the ability to manage services under the user's control with a per-user systemd instance, enabling users to start, stop, enable, and disable their own units. This is convenient for daemons and other services that are commonly run for a single user, such as mpd or to perform automated tasks like fetching mail. With some caveats it is even possible to run xorg and the entire window manager from user services.

You do not have to change that much for using systemd's user mode. Assuming you are using a recent version of systemd the user instance is started automatically per user session. You can use the ~/.config/systemd/user directory for your service and timer files. For enabling, starting and listing the timer you just have to append --user, e. g.:

$ systemctl --user enable awesome.timer

Using user mode on a server

As the systemd user instance is running per user session it might become difficult to run certain tasks automatically without an open ssh session. If you want systemd to run your user tasks without being logged in then just linger yourself with:

$ loginctl enable-linger $USER

Lingering a specific user keeps the systemd user instance running without an corresponding user session. This ensures that your tasks actually run as intended.