Skip to content

Commit 691b9eb

Browse files
Nick Simmonsindutny
Nick Simmons
authored andcommitted
fs: add recursive subdirectory support to fs.watch
Currently fs.watch does not have an option to specify if a directory should be recursively watched for events across all subdirectories. Several file watcher APIs support this. FSEvents on OS X > 10.5 is one example. libuv has added support for FSEvents, but fs.watch had no way to specify that a recursive watch was required. fs.watch now has an additional boolean option 'recursive'. When set to true, and when supported, fs.watch will return notifications for the entire directory tree hierarchy rooted at the specified path.
1 parent bae4c90 commit 691b9eb

File tree

4 files changed

+89
-10
lines changed

4 files changed

+89
-10
lines changed

doc/api/fs.markdown

+14-5
Original file line numberDiff line numberDiff line change
@@ -575,10 +575,14 @@ no-op, not an error.
575575
Watch for changes on `filename`, where `filename` is either a file or a
576576
directory. The returned object is a [fs.FSWatcher](#fs_class_fs_fswatcher).
577577

578-
The second argument is optional. The `options` if provided should be an object
579-
containing a boolean member `persistent`, which indicates whether the process
580-
should continue to run as long as files are being watched. The default is
581-
`{ persistent: true }`.
578+
The second argument is optional. The `options` if provided should be an object.
579+
The supported boolean members are `persistent` and `recursive`. `persistent`
580+
indicates whether the process should continue to run as long as files are being
581+
watched. `recursive` indicates whether all subdirectories should be watched, or
582+
only the current directory. This applies when a directory is specified, and only
583+
on supported platforms (See Caveats below).
584+
585+
The default is `{ persistent: true, recursive: false }`.
582586

583587
The listener callback gets two arguments `(event, filename)`. `event` is either
584588
'rename' or 'change', and `filename` is the name of the file which triggered
@@ -591,6 +595,10 @@ the event.
591595
The `fs.watch` API is not 100% consistent across platforms, and is
592596
unavailable in some situations.
593597

598+
The recursive option is currently supported on OS X. Only FSEvents supports this
599+
type of file watching so it is unlikely any additional platforms will be added
600+
soon.
601+
594602
#### Availability
595603

596604
<!--type=misc-->
@@ -599,7 +607,8 @@ This feature depends on the underlying operating system providing a way
599607
to be notified of filesystem changes.
600608

601609
* On Linux systems, this uses `inotify`.
602-
* On BSD systems (including OS X), this uses `kqueue`.
610+
* On BSD systems, this uses `kqueue`.
611+
* On OS X, this uses `kqueue` for files and 'FSEvents' for directories.
603612
* On SunOS systems (including Solaris and SmartOS), this uses `event ports`.
604613
* On Windows systems, this feature depends on `ReadDirectoryChangesW`.
605614

lib/fs.js

+6-3
Original file line numberDiff line numberDiff line change
@@ -1014,9 +1014,11 @@ function FSWatcher() {
10141014
}
10151015
util.inherits(FSWatcher, EventEmitter);
10161016

1017-
FSWatcher.prototype.start = function(filename, persistent) {
1017+
FSWatcher.prototype.start = function(filename, persistent, recursive) {
10181018
nullCheck(filename);
1019-
var err = this._handle.start(pathModule._makeLong(filename), persistent);
1019+
var err = this._handle.start(pathModule._makeLong(filename),
1020+
persistent,
1021+
recursive);
10201022
if (err) {
10211023
this._handle.close();
10221024
throw errnoException(err, 'watch');
@@ -1042,9 +1044,10 @@ fs.watch = function(filename) {
10421044
}
10431045

10441046
if (util.isUndefined(options.persistent)) options.persistent = true;
1047+
if (util.isUndefined(options.recursive)) options.recursive = false;
10451048

10461049
watcher = new FSWatcher();
1047-
watcher.start(filename, options.persistent);
1050+
watcher.start(filename, options.persistent, options.recursive);
10481051

10491052
if (listener) {
10501053
watcher.addListener('change', listener);

src/fs_event_wrap.cc

+5-2
Original file line numberDiff line numberDiff line change
@@ -108,12 +108,15 @@ void FSEventWrap::Start(const FunctionCallbackInfo<Value>& args) {
108108

109109
String::Utf8Value path(args[0]);
110110

111-
int err = uv_fs_event_init(wrap->env()->event_loop(), &wrap->handle_);
111+
int flags = 0;
112+
if (args[1]->IsTrue())
113+
flags |= UV_FS_EVENT_RECURSIVE;
112114

115+
int err = uv_fs_event_init(wrap->env()->event_loop(), &wrap->handle_);
113116
if (err == 0) {
114117
wrap->initialized_ = true;
115118

116-
err = uv_fs_event_start(&wrap->handle_, OnEvent, *path, 0);
119+
err = uv_fs_event_start(&wrap->handle_, OnEvent, *path, flags);
117120

118121
if (err == 0) {
119122
// Check for persistent argument
+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// Copyright Joyent, Inc. and other Node contributors.
2+
//
3+
// Permission is hereby granted, free of charge, to any person obtaining a
4+
// copy of this software and associated documentation files (the
5+
// "Software"), to deal in the Software without restriction, including
6+
// without limitation the rights to use, copy, modify, merge, publish,
7+
// distribute, sublicense, and/or sell copies of the Software, and to permit
8+
// persons to whom the Software is furnished to do so, subject to the
9+
// following conditions:
10+
//
11+
// The above copyright notice and this permission notice shall be included
12+
// in all copies or substantial portions of the Software.
13+
//
14+
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15+
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16+
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17+
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18+
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19+
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20+
// USE OR OTHER DEALINGS IN THE SOFTWARE.
21+
22+
var common = require('../common');
23+
var assert = require('assert');
24+
var path = require('path');
25+
var fs = require('fs');
26+
27+
if (process.platform === 'darwin') {
28+
var watchSeenOne = 0;
29+
30+
var testDir = common.tmpDir;
31+
32+
var filenameOne = 'watch.txt';
33+
var testsubdir = path.join(testDir, 'testsubdir');
34+
var relativePathOne = path.join('testsubdir', filenameOne);
35+
var filepathOne = path.join(testsubdir, filenameOne);
36+
37+
process.on('exit', function() {
38+
assert.ok(watchSeenOne > 0);
39+
});
40+
41+
function cleanup() {
42+
try { fs.unlinkSync(filepathOne); } catch (e) { }
43+
try { fs.rmdirSync(testsubdir); } catch (e) { }
44+
};
45+
46+
try { fs.mkdirSync(testsubdir, 0700); } catch (e) {}
47+
fs.writeFileSync(filepathOne, 'hello');
48+
49+
assert.doesNotThrow(function() {
50+
var watcher = fs.watch(testDir, {recursive: true});
51+
watcher.on('change', function(event, filename) {
52+
assert.ok('change' === event || 'rename' === event);
53+
assert.equal(relativePathOne, filename);
54+
55+
watcher.close();
56+
cleanup();
57+
++watchSeenOne;
58+
});
59+
});
60+
61+
setTimeout(function() {
62+
fs.writeFileSync(filepathOne, 'world');
63+
}, 10);
64+
}

0 commit comments

Comments
 (0)