214 lines
7.7 KiB
Diff
214 lines
7.7 KiB
Diff
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||
|
From: Ben Noordhuis <info@bnoordhuis.nl>
|
||
|
Date: Mon, 27 May 2019 14:31:53 +0200
|
||
|
Subject: fsevents: fix file event reporting
|
||
|
|
||
|
Commit 2d2af38 ("fsevents: really watch files with fsevents on macos
|
||
|
10.7+") from last November introduced a regression where events that
|
||
|
were previously reported as UV_RENAME were now reported as UV_CHANGE.
|
||
|
This commit rectifies that.
|
||
|
|
||
|
Fixes: https://github.com/nodejs/node/issues/27869
|
||
|
|
||
|
diff --git a/deps/uv/src/unix/fsevents.c b/deps/uv/src/unix/fsevents.c
|
||
|
index ddacda31fef87eee131fc2ee2ff46cc88be429d9..911023a6562319af5e9e3412ddbfd2ca14f77b05 100644
|
||
|
--- a/deps/uv/src/unix/fsevents.c
|
||
|
+++ b/deps/uv/src/unix/fsevents.c
|
||
|
@@ -49,19 +49,9 @@ void uv__fsevents_loop_delete(uv_loop_t* loop) {
|
||
|
#include <CoreFoundation/CFRunLoop.h>
|
||
|
#include <CoreServices/CoreServices.h>
|
||
|
|
||
|
-/* These are macros to avoid "initializer element is not constant" errors
|
||
|
+/* Macro to avoid "initializer element is not constant" errors
|
||
|
* with old versions of gcc.
|
||
|
*/
|
||
|
-#define kFSEventsModified (kFSEventStreamEventFlagItemFinderInfoMod | \
|
||
|
- kFSEventStreamEventFlagItemModified | \
|
||
|
- kFSEventStreamEventFlagItemInodeMetaMod | \
|
||
|
- kFSEventStreamEventFlagItemChangeOwner | \
|
||
|
- kFSEventStreamEventFlagItemXattrMod)
|
||
|
-
|
||
|
-#define kFSEventsRenamed (kFSEventStreamEventFlagItemCreated | \
|
||
|
- kFSEventStreamEventFlagItemRemoved | \
|
||
|
- kFSEventStreamEventFlagItemRenamed)
|
||
|
-
|
||
|
#define kFSEventsSystem (kFSEventStreamEventFlagUserDropped | \
|
||
|
kFSEventStreamEventFlagKernelDropped | \
|
||
|
kFSEventStreamEventFlagEventIdsWrapped | \
|
||
|
@@ -289,8 +279,6 @@ static void uv__fsevents_event_cb(ConstFSEventStreamRef streamRef,
|
||
|
path--;
|
||
|
len++;
|
||
|
}
|
||
|
- /* Created and Removed seem to be always set, but don't make sense */
|
||
|
- flags &= ~kFSEventsRenamed;
|
||
|
} else {
|
||
|
/* Skip forward slash */
|
||
|
path++;
|
||
|
@@ -311,12 +299,12 @@ static void uv__fsevents_event_cb(ConstFSEventStreamRef streamRef,
|
||
|
|
||
|
memset(event, 0, sizeof(*event));
|
||
|
memcpy(event->path, path, len + 1);
|
||
|
- event->events = UV_RENAME;
|
||
|
+ event->events = UV_CHANGE;
|
||
|
|
||
|
- if (0 == (flags & kFSEventsRenamed)) {
|
||
|
- if (0 != (flags & kFSEventsModified) ||
|
||
|
- 0 == (flags & kFSEventStreamEventFlagItemIsDir))
|
||
|
- event->events = UV_CHANGE;
|
||
|
+ if ((flags & kFSEventStreamEventFlagItemIsDir) ||
|
||
|
+ (flags & kFSEventStreamEventFlagItemRemoved) ||
|
||
|
+ (flags & kFSEventStreamEventFlagItemRenamed)) {
|
||
|
+ event->events = UV_RENAME;
|
||
|
}
|
||
|
|
||
|
QUEUE_INSERT_TAIL(&head, &event->member);
|
||
|
diff --git a/deps/uv/test/test-fs-event.c b/deps/uv/test/test-fs-event.c
|
||
|
index ea34bd63a70625c3e2c60d5a1bbb087c5f0bbb2e..38d722a27ef1a78717726272d54578c734280242 100644
|
||
|
--- a/deps/uv/test/test-fs-event.c
|
||
|
+++ b/deps/uv/test/test-fs-event.c
|
||
|
@@ -62,6 +62,15 @@ static char fs_event_filename[1024];
|
||
|
static int timer_cb_touch_called;
|
||
|
static int timer_cb_exact_called;
|
||
|
|
||
|
+static void expect_filename(const char* path, const char* expected) {
|
||
|
+#if defined(__APPLE__) || defined(_WIN32) || defined(__linux__)
|
||
|
+ ASSERT(0 == strcmp(path, expected));
|
||
|
+#else
|
||
|
+ if (path != NULL)
|
||
|
+ ASSERT(0 == strcmp(path, expected));
|
||
|
+#endif
|
||
|
+}
|
||
|
+
|
||
|
static void fs_event_fail(uv_fs_event_t* handle,
|
||
|
const char* filename,
|
||
|
int events,
|
||
|
@@ -130,11 +139,7 @@ static void fs_event_cb_dir(uv_fs_event_t* handle, const char* filename,
|
||
|
ASSERT(handle == &fs_event);
|
||
|
ASSERT(status == 0);
|
||
|
ASSERT(events == UV_CHANGE);
|
||
|
- #if defined(__APPLE__) || defined(_WIN32) || defined(__linux__)
|
||
|
- ASSERT(strcmp(filename, "file1") == 0);
|
||
|
- #else
|
||
|
- ASSERT(filename == NULL || strcmp(filename, "file1") == 0);
|
||
|
- #endif
|
||
|
+ expect_filename(filename, "file1");
|
||
|
ASSERT(0 == uv_fs_event_stop(handle));
|
||
|
uv_close((uv_handle_t*)handle, close_cb);
|
||
|
}
|
||
|
@@ -312,11 +317,7 @@ static void fs_event_cb_file(uv_fs_event_t* handle, const char* filename,
|
||
|
ASSERT(handle == &fs_event);
|
||
|
ASSERT(status == 0);
|
||
|
ASSERT(events == UV_CHANGE);
|
||
|
- #if defined(__APPLE__) || defined(_WIN32) || defined(__linux__)
|
||
|
- ASSERT(strcmp(filename, "file2") == 0);
|
||
|
- #else
|
||
|
- ASSERT(filename == NULL || strcmp(filename, "file2") == 0);
|
||
|
- #endif
|
||
|
+ expect_filename(filename, "file2");
|
||
|
ASSERT(0 == uv_fs_event_stop(handle));
|
||
|
uv_close((uv_handle_t*)handle, close_cb);
|
||
|
}
|
||
|
@@ -339,11 +340,7 @@ static void fs_event_cb_file_current_dir(uv_fs_event_t* handle,
|
||
|
ASSERT(handle == &fs_event);
|
||
|
ASSERT(status == 0);
|
||
|
ASSERT(events == UV_CHANGE);
|
||
|
- #if defined(__APPLE__) || defined(_WIN32) || defined(__linux__)
|
||
|
- ASSERT(strcmp(filename, "watch_file") == 0);
|
||
|
- #else
|
||
|
- ASSERT(filename == NULL || strcmp(filename, "watch_file") == 0);
|
||
|
- #endif
|
||
|
+ expect_filename(filename, "watch_file");
|
||
|
|
||
|
/* Regression test for SunOS: touch should generate just one event. */
|
||
|
{
|
||
|
@@ -619,6 +616,69 @@ TEST_IMPL(fs_event_watch_file_exact_path) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
+static void file_remove_cb(uv_fs_event_t* handle,
|
||
|
+ const char* path,
|
||
|
+ int events,
|
||
|
+ int status) {
|
||
|
+ fs_event_cb_called++;
|
||
|
+
|
||
|
+ expect_filename(path, "file1");
|
||
|
+ /* TODO(bnoordhuis) Harmonize the behavior across platforms. Right now
|
||
|
+ * this test merely ensures the status quo doesn't regress.
|
||
|
+ */
|
||
|
+#if defined(_AIX) || defined(__linux__)
|
||
|
+ ASSERT(UV_CHANGE == events);
|
||
|
+#else
|
||
|
+ ASSERT(UV_RENAME == events);
|
||
|
+#endif
|
||
|
+ ASSERT(0 == status);
|
||
|
+
|
||
|
+ uv_close((uv_handle_t*) handle, NULL);
|
||
|
+}
|
||
|
+
|
||
|
+static void file_remove_next(uv_timer_t* handle) {
|
||
|
+ uv_close((uv_handle_t*) handle, NULL);
|
||
|
+ remove("watch_dir/file1");
|
||
|
+}
|
||
|
+
|
||
|
+static void file_remove_start(uv_timer_t* handle) {
|
||
|
+ uv_fs_event_t* watcher;
|
||
|
+ uv_loop_t* loop;
|
||
|
+
|
||
|
+ loop = handle->loop;
|
||
|
+ watcher = handle->data;
|
||
|
+
|
||
|
+ ASSERT(0 == uv_fs_event_init(loop, watcher));
|
||
|
+ ASSERT(0 == uv_fs_event_start(watcher, file_remove_cb, "watch_dir/file1", 0));
|
||
|
+ ASSERT(0 == uv_timer_start(handle, file_remove_next, 50, 0));
|
||
|
+}
|
||
|
+
|
||
|
+TEST_IMPL(fs_event_watch_file_remove) {
|
||
|
+ uv_fs_event_t watcher;
|
||
|
+ uv_timer_t timer;
|
||
|
+ uv_loop_t* loop;
|
||
|
+
|
||
|
+#if defined(__MVS__)
|
||
|
+ RETURN_SKIP("test does not work on this OS");
|
||
|
+#endif
|
||
|
+
|
||
|
+ remove("watch_dir/file1");
|
||
|
+ remove("watch_dir/");
|
||
|
+ create_dir("watch_dir");
|
||
|
+ create_file("watch_dir/file1");
|
||
|
+
|
||
|
+ loop = uv_default_loop();
|
||
|
+ timer.data = &watcher;
|
||
|
+
|
||
|
+ ASSERT(0 == uv_timer_init(loop, &timer));
|
||
|
+ ASSERT(0 == uv_timer_start(&timer, file_remove_start, 500, 0));
|
||
|
+ ASSERT(0 == uv_run(loop, UV_RUN_DEFAULT));
|
||
|
+ ASSERT(1 == fs_event_cb_called);
|
||
|
+
|
||
|
+ MAKE_VALGRIND_HAPPY();
|
||
|
+ return 0;
|
||
|
+}
|
||
|
+
|
||
|
TEST_IMPL(fs_event_watch_file_twice) {
|
||
|
#if defined(NO_FS_EVENTS)
|
||
|
RETURN_SKIP(NO_FS_EVENTS);
|
||
|
diff --git a/deps/uv/test/test-list.h b/deps/uv/test/test-list.h
|
||
|
index 8886b07c8a74736208d5614257f832f59d79633a..3e758a1d2cbe104a691d362411770006d1494bae 100644
|
||
|
--- a/deps/uv/test/test-list.h
|
||
|
+++ b/deps/uv/test/test-list.h
|
||
|
@@ -333,6 +333,7 @@ TEST_DECLARE (fs_event_watch_dir_short_path)
|
||
|
#endif
|
||
|
TEST_DECLARE (fs_event_watch_file)
|
||
|
TEST_DECLARE (fs_event_watch_file_exact_path)
|
||
|
+TEST_DECLARE (fs_event_watch_file_remove)
|
||
|
TEST_DECLARE (fs_event_watch_file_twice)
|
||
|
TEST_DECLARE (fs_event_watch_file_current_dir)
|
||
|
#ifdef _WIN32
|
||
|
@@ -920,6 +921,7 @@ TASK_LIST_START
|
||
|
#endif
|
||
|
TEST_ENTRY (fs_event_watch_file)
|
||
|
TEST_ENTRY (fs_event_watch_file_exact_path)
|
||
|
+ TEST_ENTRY (fs_event_watch_file_remove)
|
||
|
TEST_ENTRY (fs_event_watch_file_twice)
|
||
|
TEST_ENTRY (fs_event_watch_file_current_dir)
|
||
|
#ifdef _WIN32
|