Skip to content

Commit ed85c95

Browse files
committed
doc,test: documents behaviour of non-existent file
As per the discussion in #2093 (comment), this patch documents the behavior of calling fs.watchFile() with a path that does not yet exist. This patch also includes a test which checks if a file not present, the callback is invoked at least once and if the file is created after the callback is invoked, it will be invoked again with new stat objects. PR-URL: #2169 Reviewed-By: Ben Noordhuis <[email protected]> Reviewed-By: Brendan Ashworth <[email protected]>
1 parent ea05e76 commit ed85c95

File tree

2 files changed

+61
-6
lines changed

2 files changed

+61
-6
lines changed

doc/api/fs.markdown

+4-1
Original file line numberDiff line numberDiff line change
@@ -577,7 +577,10 @@ If you want to be notified when the file was modified, not just accessed
577577
you need to compare `curr.mtime` and `prev.mtime`.
578578

579579
_Note: when an `fs.watchFile` operation results in an `ENOENT` error, it will
580-
invoke the callback once. This is a change in functionality since v0.10._
580+
invoke the listener once, with all the fields zeroed (or, for dates, the Unix
581+
Epoch). In Windows, `blksize` and `blocks` fields will be `undefined`, instead
582+
of zero. If the file is created later on, the listener will be called again,
583+
with the latest stat objects. This is a change in functionality since v0.10._
581584

582585
_Note: `fs.watch` is more efficient than `fs.watchFile` and `fs.unwatchFile`.
583586
`fs.watch` should be used instead of `fs.watchFile` and `fs.unwatchFile`

test/parallel/test-fs-watchfile.js

+57-5
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
11
'use strict';
22

3+
const common = require('../common');
34
const fs = require('fs');
45
const path = require('path');
56
const assert = require('assert');
6-
const common = require('../common');
77
const fixtures = path.join(__dirname, '..', 'fixtures');
88

99
// Basic usage tests.
@@ -19,8 +19,60 @@ assert.throws(function() {
1919
fs.watchFile(new Object(), function() {});
2020
}, /Path must be a string/);
2121

22-
// Test ENOENT. Should fire once.
23-
const enoentFile = path.join(fixtures, 'empty', 'non-existent-file');
22+
const enoentFile = path.join(fixtures, 'non-existent-file');
23+
const expectedStatObject = new fs.Stats(
24+
0, // dev
25+
0, // mode
26+
0, // nlink
27+
0, // uid
28+
0, // gid
29+
0, // rdev
30+
common.isWindows ? undefined : 0, // blksize
31+
0, // ino
32+
0, // size
33+
common.isWindows ? undefined : 0, // blocks
34+
Date.UTC(1970, 0, 1, 0, 0, 0), // atime
35+
Date.UTC(1970, 0, 1, 0, 0, 0), // mtime
36+
Date.UTC(1970, 0, 1, 0, 0, 0), // ctime
37+
Date.UTC(1970, 0, 1, 0, 0, 0) // birthtime
38+
);
39+
40+
function removeTestFile() {
41+
try {
42+
fs.unlinkSync(enoentFile);
43+
} catch (ex) {
44+
if (ex.code !== 'ENOENT') {
45+
throw ex;
46+
}
47+
}
48+
}
49+
50+
// Make sure that the file does not exist, when the test starts
51+
removeTestFile();
52+
53+
// If the file initially didn't exist, and gets created at a later point of
54+
// time, the callback should be invoked again with proper values in stat object
55+
var fileExists = false;
56+
2457
fs.watchFile(enoentFile, common.mustCall(function(curr, prev) {
25-
fs.unwatchFile(enoentFile);
26-
}));
58+
if (!fileExists) {
59+
// If the file does not exist, all the fields should be zero and the date
60+
// fields should be UNIX EPOCH time
61+
assert.deepStrictEqual(curr, expectedStatObject);
62+
assert.deepStrictEqual(prev, expectedStatObject);
63+
// Create the file now, so that the callback will be called back once the
64+
// event loop notices it.
65+
fs.closeSync(fs.openSync(enoentFile, 'w'));
66+
fileExists = true;
67+
} else {
68+
// If the ino (inode) value is greater than zero, it means that the file is
69+
// present in the filesystem and it has a valid inode number.
70+
assert(curr.ino > 0);
71+
// As the file just got created, previous ino value should be lesser than
72+
// or equal to zero (non-existent file).
73+
assert(prev.ino <= 0);
74+
// Stop watching the file and delete it
75+
fs.unwatchFile(enoentFile);
76+
removeTestFile();
77+
}
78+
}, 2));

0 commit comments

Comments
 (0)