From 4b377e5c57f50eb122bda8c22808ac39ebd9f6ad Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Fri, 18 Nov 2022 00:26:18 +0200 Subject: [PATCH] fs: fix `nonNativeWatcher` leak of `StatWatchers` PR-URL: https://github.com/nodejs/node/pull/45501 Reviewed-By: Yagiz Nizipli --- lib/internal/fs/recursive_watch.js | 9 +++----- test/parallel/test-fs-watch-recursive.js | 27 ++++++++++++++++++++++++ 2 files changed, 30 insertions(+), 6 deletions(-) diff --git a/lib/internal/fs/recursive_watch.js b/lib/internal/fs/recursive_watch.js index ed146fa28bed8d..5c7a3e5b39a9a2 100644 --- a/lib/internal/fs/recursive_watch.js +++ b/lib/internal/fs/recursive_watch.js @@ -154,14 +154,11 @@ class FSWatcher extends EventEmitter { this.#symbolicFiles.add(f); } + this.#files.set(f, file); if (file.isFile()) { this.#watchFile(f); - } else { - this.#files.set(f, file); - - if (file.isDirectory() && !file.isSymbolicLink()) { - await this.#watchFolder(f); - } + } else if (file.isDirectory() && !file.isSymbolicLink()) { + await this.#watchFolder(f); } } } diff --git a/test/parallel/test-fs-watch-recursive.js b/test/parallel/test-fs-watch-recursive.js index 70b413814e78f4..acf3a3bc6a421d 100644 --- a/test/parallel/test-fs-watch-recursive.js +++ b/test/parallel/test-fs-watch-recursive.js @@ -1,6 +1,7 @@ 'use strict'; const common = require('../common'); +const { setTimeout } = require('timers/promises'); const assert = require('assert'); @@ -52,3 +53,29 @@ if (common.isOSX) { process.on('exit', function() { assert(watcherClosed, 'watcher Object was not closed'); }); + +(async () => { + // Assert recursive watch does not leak handles + const rootDirectory = fs.mkdtempSync(testDir + path.sep); + const testDirectory = path.join(rootDirectory, 'test-7'); + const filePath = path.join(testDirectory, 'only-file.txt'); + fs.mkdirSync(testDirectory); + + let watcherClosed = false; + const watcher = fs.watch(testDirectory, { recursive: true }); + watcher.on('change', common.mustCallAtLeast(async (event, filename) => { + await setTimeout(common.platformTimeout(100)); + if (filename === path.basename(filePath)) { + watcher.close(); + watcherClosed = true; + } + await setTimeout(common.platformTimeout(100)); + assert(!process._getActiveHandles().some((handle) => handle.constructor.name === 'StatWatcher')); + })); + + process.on('exit', function() { + assert(watcherClosed, 'watcher Object was not closed'); + }); + await setTimeout(common.platformTimeout(100)); + fs.writeFileSync(filePath, 'content'); +})().then(common.mustCall());