Monitor file and directory activity with incron
Controlled Files
The incron utility provides an easy way to initiate commands and scripts triggered by filesystem events.
If you want to perform a task every time a specific event occurs, you could employ various techniques for polling or log watching; however, Linux provides a more general solution in the form of a tool called incron. Incron uses the inotify subsystem [1] to listen for events that affect a filesystem, such as opening, creating, or deleting a file; accessing a directory; or changing an attribute.
The name incron suggests the common cron utility [2]; however, cron jobs are triggered by a moment in time (every Friday, once a day at 3 am, in August, etc.), whereas incron is triggered by file or directory events. The gears in your head are probably already spinning, thinking of all the potential uses of incron. So, in this article, I describe how to set up incron and put it to work in some simple examples.
Getting incron
The incron tool set is not usually preinstalled by default in most distros, but because it is an implementation of the inotify subsystem (see the box "The Origin of incron"), it is likely in your distro's repository, so you can install it through the package manager. However, if for some reason you can't install the necessary packages from your distro's repos, you can download the latest version [3] and build from source.
The Origin of incron
The in in incron comes from inode, the data structure that contains the details (location on the physical disk, size, owner, etc.) of each file and directory in the filesystem. Inotify is the kernel subsystem on which incron is based. This subsystem monitors file and directory changes and can be queried by applications, allowing, for example, a new folder icon to pop up in real time in a graphical file browser window (e.g., Nautilus or Dolphin), even if the underlying directory has been created by some other means (e.g., with mkdir
from a shell). It can also be used to inform an application that an open file has been changed on disk by some other program, thus allowing the user to overwrite or rename the version being worked on.
Note that incron is a relatively recent technology – the inotify notification framework was first implemented in kernel 2.6.13 – so if you're running an older kernel, incron will not work.
Incron comprises several bits and pieces, of which the main component is the incrond
daemon. This daemon installs into /etc/init.d
. Depending on your system, it can be started with one of the following commands
# /etc/init.d/incrond start # /etc/init.d/incron start
or, if you're using systemd,
# systemctl status incron.service
or any variation thereof.
Hello incron!
Once the daemon is running, you can create your own particular "Hello World!" In this first project, I'll create a logfile and add a line to it every time a file is added to a monitored directory.
First, I create the directory I want to monitor:
$ mkdir my_dir
The incron system is similar to cron in that it saves a file for each user. This file, called incrontab
, contains the events you want to monitor linked to the actions you want to run. Although you could edit this file by hand (I'll look at that a bit later), the done way of doing things is to use the incrontab
instruction with, in this case, the -e
(for edit) argument:
incrontab -e
This command opens the file for the current user in the preconfigured default editor – that is, whatever your $EDITOR
environment variable points to (or vi if it isn't pointing to anything). As with cron, each event/task pair has to be on one line and comprises:
<a path>
– the path to the directory or file to monitor.<a mask>
– the events to monitor (see Table 1).<a command>
– the action to run. It can be an individual command or a script that runs from the shell.
Table 1
Monitored Events
Event | Meaning |
---|---|
Common Events |
|
|
File was accessed (read) |
|
Metadata was changed (permissions, timestamps, extended attributes, etc.) |
|
File opened for writing was closed |
|
File not opened for writing was closed |
|
Combines |
|
File/directory created in watched directory |
|
File/directory deleted from watched directory |
|
Watched file/directory was deleted |
|
File was modified |
|
Watched file/directory was moved |
|
File moved out of watched directory |
|
File moved into watched directory |
|
A combination of |
|
File was opened |
Special Events |
|
|
Combines all of the above events |
|
Don't dereference pathname if it is a symbolic link |
|
Monitor pathname for only one event |
|
Only watch pathname if it is a directory |
Wildcard Event |
|
|
Disable monitoring of events until the current event is handled completely (until its child process exits – avoids infinite loops) |
In this example, incrontab will contain a single line:
/home/<user>/my_dir IN_CREATE /home/<user>/bin/hello.sh
The hello.sh
file contains the very short script,
#!/bin/bash echo "File created." >> /home/<user>/file_log
where <user> is the username of interest. For this command to work, you must make hello.sh
executable with:
chmod a+x bin/
Once you have saved and closed the user's incrontab file, the program will show the message table updated, indicating the changes have been registered. Now, if you create a file in my_dir
by typing, for example,
$ touch my_dir/file1.txt
the file_log
file will appear in your home directory containing the line File created.
Note the first limitation of incron: Compound operations, such as the echo
instruction that redirects its output to a file in hello.sh
, must be wrapped in shell script, or incron will choke. Because incron is not a shell command interpreter, it does not understand such things as the redirections, pipes, and globbings that make up a compound instruction. The only thing incron can use in the command section of the line is the instruction's name and its parameters. For example, this command
/home/paul/my_dir IN_CREATE echo "File created." >> /home/paul/file_log
would not work as expected, because it would be interpreted as follows:
echo
is the instruction to run (correct)."File created."
is a parameter forecho
(also correct).>>
is another argument forecho
(incorrect)./home/paul/file_log
is also an argument forecho
(also incorrect).
This behavior is the default for incron, so it won't generate any errors in its logs. (You can track incron's errors or lack thereof by checking /var/log/cron
.) Incron simply ignores what it doesn't understand and carries on.
This example also illustrates the second limitation of incron: In the incrontab line, you have to specify the complete absolute path to the executable that, in this case, lives in the user's own bin/
directory. Although this directory might be included in the user's own $PATH
environment variable, the incron daemon runs as superuser, so the executable would have to be in the root's $PATH
for the daemon to find it; otherwise, you have to specify the absolute path. Thus, you must make sure your executable is in /bin
, /usr/bin
, /usr/local/bin
, or somewhere equally accessible, or point to the program with an absolute path.
Wild Cards
This "Hello World" incron program is as silly as any other, but with a couple of small changes you can modify it to make it much more useful. For example, registering the name of a created file in the logfile is easy. Incron provides a series of default variables (for some reason called "wildcards" in incron jargon) that contain this kind of information. (See Table 2 for a full list.) For the time being, $#
, which contains the name of the file affected by the event, is of interest.
Table 2
Wildcards
Wildcard | Meaning |
---|---|
|
Dollar sign |
|
Watched filesystem path |
|
Event-related file name |
|
Event flag (textual) |
|
Event flag (numerical) |
Open your incrontab file again (incrontab -e
) and change your monitoring line to:
/home/paul/my_dir IN_CREATE /home/paul/bin/hello.sh $#
Next, modify your hello.sh
script to:
#!/bin/bash echo "File $1 created." >> /home/paul/file_log
Now try it out by creating a new file in my_dir
:
$ touch my_dir/file2.txt
If you look in file_log
, you'll see that it contains a new line that says File file2.txt created..
Buy this article as PDF
(incl. VAT)
Buy Linux Magazine
Subscribe to our Linux Newsletters
Find Linux and Open Source Jobs
Subscribe to our ADMIN Newsletters
Support Our Work
Linux Magazine content is made possible with support from readers like you. Please consider contributing when you’ve found an article to be beneficial.
News
-
Gnome 48 Debuts New Audio Player
To date, the audio player found within the Gnome desktop has been meh at best, but with the upcoming release that all changes.
-
Plasma 6.3 Ready for Public Beta Testing
Plasma 6.3 will ship with KDE Gear 24.12.1 and KDE Frameworks 6.10, along with some new and exciting features.
-
Budgie 10.10 Scheduled for Q1 2025 with a Surprising Desktop Update
If Budgie is your desktop environment of choice, 2025 is going to be a great year for you.
-
Firefox 134 Offers Improvements for Linux Version
Fans of Linux and Firefox rejoice, as there's a new version available that includes some handy updates.
-
Serpent OS Arrives with a New Alpha Release
After months of silence, Ikey Doherty has released a new alpha for his Serpent OS.
-
HashiCorp Cofounder Unveils Ghostty, a Linux Terminal App
Ghostty is a new Linux terminal app that's fast, feature-rich, and offers a platform-native GUI while remaining cross-platform.
-
Fedora Asahi Remix 41 Available for Apple Silicon
If you have an Apple Silicon Mac and you're hoping to install Fedora, you're in luck because the latest release supports the M1 and M2 chips.
-
Systemd Fixes Bug While Facing New Challenger in GNU Shepherd
The systemd developers have fixed a really nasty bug amid the release of the new GNU Shepherd init system.
-
AlmaLinux 10.0 Beta Released
The AlmaLinux OS Foundation has announced the availability of AlmaLinux 10.0 Beta ("Purple Lion") for all supported devices with significant changes.
-
Gnome 47.2 Now Available
Gnome 47.2 is now available for general use but don't expect much in the way of newness, as this is all about improvements and bug fixes.