fswatch
is a file change monitor that receives notifications when the contents
of the specified files or directories are modified. fswatch
implements four
kinds of monitors:
- A monitor based on the File System Events API of Apple OS X.
- A monitor based on kqueue, an event notification interface introduced in FreeBSD 4.1 and supported on most *BSD systems (including OS X).
- A monitor based on inotify, a Linux kernel subsystem that reports file system changes to applications.
- A monitor which periodically stats the file system, saves file modification
times in memory and manually calculates file system changes, which can work
on any operating system where
stat (2)
can be used.
fswatch
should build and work correctly on any system shipping either of the
aforementioned APIs.
Alan Dipert wrote the first implementation of fswatch
in 2009. This
version ran exclusively on OS X and relied on the FSEvents API to get
change events from the OS.
At the end of 2013 Enrico M. Crisostomo
wrote fsw
aiming at providing not only a drop-in replacement for fswatch
,
but a common front-end from multiple file system change events APIs, including:
- OS X FSEvents.
- *BSD kqueue.
- Linux inotify.
In April 2014 Alan and Enrico, in the best interest of users of either
fswatch
and fsw
, agreed on merging the two programs together. At the same
time, Enrico was taking over fswatch
as a maintainer.
As a consequence, development of fswatch
will continue on its main
repository while the fsw
repository will likely be frozen and
its documentation updated to redirect users to fswatch
.
fswatch
main features are:
- Support for many OS-specific APIs such as kevent, inotify and FSEvents.
- Recursive directory monitoring.
- Path filtering using including and excluding regular expressions.
- Customizable record format.
The limitations of fswatch
depend largely on the monitor being used:
- The FSEvents monitor, available only on Apple OS X, has no known limitations and scales very well with the number of files being observed.
- The kqueue monitor, available on any *BSD system featuring kqueue, requires
a file descriptor to be opened for every file being watched. As a result,
this monitor scales badly with the number of files being observed and may
begin to misbehave as soon as the
fswatch
process runs out of file descriptors. In this case,fswatch
dumps one error on standard error for every file that cannot be opened. - The inotify monitor, available on Linux since kernel 2.6.13, may suffer a
queue overflow if events are generated faster than they are read from the
queue. In any case, the application is guaranteed to receive an overflow
notification which can be handled to gracefully recover.
fswatch
currently throws an exception if a queue overflow occurs. Future versions will handle the overflow by emitting proper notifications. - The poll monitor, available on any platform, only relies on available CPU and memory to perform its task. The performance of this monitor degrades linearly with the number of files being watched.
Usage recommendations are as follows:
- On OS X, use only the FSEvents monitor (which is the default behaviour).
- On Linux, use the inotify monitor (which is the default behaviour).
- If the number of files to observe is sufficiently small, use the kqueue monitor. Beware that on some systems the maximum number of file descriptors that can be opened by a process is set to a very low value (values as low as 256 are not uncommon), even if the operating system may allow a much larger value. In this case, check your OS documentation to raise this limit on either a per process or a system-wide basis.
- If feasible, watch directories instead of watching files. Properly crafting the receiving side of the events to deal with directories may sensibly reduce the monitor resource consumption.
- If none of the above applies, use the poll monitor. The authors' experience
indicates that
fswatch
requires approximately 150 MB or RAM memory to observe a hierarchy of 500.000 files with a minimum path length of 32 characters. A common bottleneck of the poll monitor is disk access, sincestat()
-ing a great number of files may take a huge amount of time. In this case, the latency should be set to a sufficiently large value in order to reduce the performance degradation that may result from frequent disk access.
A regular user may be able to fetch fswatch
from the package manager of your
OS or a third-party one. If you are looking for fswatch
for OS X, you can
install it using either MacPorts or Homebrew:
# MacPorts
$ port install fswatch
# Homebrew
$ brew install fswatch
Check your favourite package manager and let us know if fswatch
is missing
there.
A user who whishes to build fswatch
should get a release tarball.
A release tarball contains everything a user needs to build fswatch
on his
system, following the instructions detailed in the Installation section below
and the INSTALL file.
A developer who wishes to modify fswatch
should get the sources (either from
a source tarball or cloning the repository) and have the GNU Build System
installed on his machine. Please read README.gnu-build-system
to get further
details about how to bootstrap fswatch
from sources on your machine.
Getting a copy of the source repository is not recommended unless you are a developer, you have the GNU Build System installed on your machine and you know how to bootstrap it on the sources.
See the INSTALL
file for detailed information about how to configure and
install fswatch
. Since the fswatch
builds and uses dynamic libraries, in
some platforms you may need to perform additional tasks before you can use
fswatch
:
-
Make sure the installation directory of dynamic libraries ($PREFIX/lib) is included in the lookup paths of the dynamic linker of your operating system. The default path,
/usr/local/lib
, will work in nearly every operating system. -
Refreshing the links and cache to the dynamic libraries may be required. In GNU/Linux systems you may need to run
ldconfig
:$ ldconfig
fswatch
is a C++ program and a C++ compiler compliant with the C++11 standard
is required to compile it. Check your OS documentation for information about
how to install the C++ toolchain and the C++ runtime.
No other software packages or dependencies are required to configure and
install fswatch
but the aforementioned APIs used by the file system monitors.
fswatch
is localizable and internally uses GNU gettext
to decouple
localizable string from their translation. The currently available locales
are:
- English (
en
). - Italian (
it
). - Spanish (
es
).
To build fswatch
with localization support, you need to have gettext
installed on your system. If configure
cannot find <libintl.h>
or the
linker cannot find libintl
, then you may need to manually provide their
location to configure
, usually using the CPPFLAGS
and the LDFLAGS
variables. See README.osx
for an example.
If gettext
is not available on your system, fswatch
shall build correctly,
but it will lack localization support and the only available locale will be
English.
fswatch
provides the following documentation:
- Texinfo documentation, included with the distribution.
- A wiki page.
- A man page.
fswatch
official documentation is provided in Texinfo format.
This is the most comprehensive source of information about fswatch
.
The man page, in particular, is a stub meant for quick reference from the
command line and is not guaranteed to be kept up to date.
By default, only Info and DVI formats are generated when fswatch
is built,
according to the default rules of the GNU Build Systems. However, a PDF manual
typeset with TeX can be generated from the Texinfo sources by issuing this
command:
$ make pdf
and can then be installed invoking the install-pdf
target:
$ make install-pdf
Since typical users will not have a TeX distribution installed in their
computers, the PDF manuals for every version of fswatch
will be hosted at
this address.
If you are installing fswatch
using a package manager and you would like the
PDF manual to be bundled into the package, please send a feature request to the
package maintainer.
The previous major version of fswatch
(v. 0.x) allowed users to run a command
whenever a set of changes was detected with the following syntax:
$ fswatch path program
Starting with fswatch
v. 1.x this behaviour is no longer supported. The
rationale behind this decision includes:
- The old version only allows watching one path.
- The command to execute was passed as last argument, alongside the path to watch, making it difficult to extend the program functionality to add multiple path support
- The old version forks and executes
/bin/bash
, which is neither portable, nor guaranteed to succeed, nor desirable by users of other shells. - No information about the change events is passed to the forked process.
To solve the aforementioned issues and keep fswatch
consistent with common
UNIX practices, the behaviour has changed and fswatch
now prints event
records to the standard output that users can process further by piping the
output of fswatch
to other programs.
To fully support the old use, the -o/--one-per-batch
option was added in
v. 1.3.3. When specified, fswatch
will only dump 1 event to standard output
which can be used to trigger another program
:
$ fswatch -o path | xargs -n1 program
In this case, program
will receive the number of change events as first
argument. If no argument should be passed to program
, then the following
command could be used:
$ fswatch -o path | xargs -n1 -I{} program
Although we encourage you to embrace the new fswatch behaviour and update your
scripts, we provide a little wrapper called fswatch-run
which is installed
alongside fswatch
which lets you use the legacy syntax:
$ fswatch-run path [paths] program
Under the hood, fswatch-run
simply calls fswatch -o
piping its output to
xargs
.
fswatch-run
is a symbolic link to a shell-specific wrapper. Currently, ZSH
and Bash scripts are provided. If no suitable shells are found in the target
system, the fswatch-run
symbolic link is not created.
fswatch
accepts a list of paths for which change events should be received:
$ fswatch [options] ... path-0 ... path-n
The event stream is created even if any of the paths do not exist yet. If they
are created after fswatch
is launched, change events will be properly
received. Depending on the watcher being used, newly created paths will be
monitored after the amount of configured latency has elapsed.
The output of fswatch
can be piped to other program in order to process it
further:
$ fswatch -0 path | while read -d "" event \
do \
// do something with ${event}
done
To run a command when a set of change events is printed to standard output but no event details are required, then the following command can be used:
$ fswatch -o path | xargs -n1 -I{} program
The behaviour is consistent with earlier versions of fswatch
(v. 0.x).
Please, read the Compatibility Issues with fswatch v. 0.x section for further
information.
By default fswatch
chooses the best monitor available on the current
platform, in terms of performance and resource consumption. If the user wishes
to specify a different monitor, the -m
option can be used to specify the
monitor by name:
$ fswatch -m kqueue_monitor path
The list of available monitors can be obtained with the -h
option.
For more information, refer to the fswatch
man page.
Bug reports can be sent directly to the authors.
Copyright (C) 2014 Enrico M. Crisostomo
This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version.
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with this program. If not, see http://www.gnu.org/licenses/.