How to prevent ReadDirectoryChangesW from returning spurious Modified events when iterating directory

158 Views Asked by At

I use ReadDirectoryChangesW() to detect changes made to a directory tree. When testing my implementation, I found that iterating over a just created directory tree causes ReadDirectoryChangesW() to return spurious Modified events.

  • Create a directory tree
  • Start watching the tree with ReadDirectoryChangesW()
  • Iterate over the directory tree

Expected: ReadDirectoryChangesW() reports no changes.

Actual: ReadDirectoryChangesW() reports Modified changes.

I used the following code to reproduce this issue:

#include <windows.h>
#include <filesystem>
#include <iostream>
#include <string>
#include <memory>

namespace
{
    HANDLE createHandle(std::filesystem::path const& directory) {
        std::wstring path = directory.wstring();
        LPCWSTR dirPath = static_cast<LPCWSTR>(path.c_str());
        HANDLE handle = CreateFileW(
            dirPath,
            FILE_LIST_DIRECTORY,
            FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
            NULL,
            OPEN_EXISTING,
            FILE_FLAG_BACKUP_SEMANTICS | FILE_FLAG_OVERLAPPED,
            NULL);
        return handle;
    }

    class DirectoryWatcher
    {
    private:
        std::filesystem::path _rootDir;

        HANDLE _dirHandle;
        DWORD _bufferSizeInBytes;
        std::unique_ptr<DWORD> _buffer;
        OVERLAPPED _overlapped;

        std::mutex _mutex;
        std::condition_variable _cond;
        bool _running;
        std::atomic<bool> _stop;
        std::unique_ptr<std::thread> _watcher;

    public:
        DirectoryWatcher(std::filesystem::path const& directory)
            : _rootDir(directory)
            , _dirHandle(createHandle(_rootDir))
            , _bufferSizeInBytes(32 * 1024)
            , _buffer(new DWORD[_bufferSizeInBytes / sizeof(DWORD)])
            , _running(false)
            , _stop(false)
        {
            if (_dirHandle == INVALID_HANDLE_VALUE) throw std::runtime_error("dir handle creation failed");
            memset(&_overlapped, 0, sizeof(OVERLAPPED));
            _overlapped.hEvent = CreateEvent(NULL, FALSE, 0, NULL);
            if (_overlapped.hEvent == INVALID_HANDLE_VALUE) throw std::runtime_error("event handle creation failed");
            _watcher = std::make_unique<std::thread>(&DirectoryWatcher::run, this);

            std::unique_lock<std::mutex> lock(_mutex);
            while (!_running) _cond.wait(lock);
        }

        ~DirectoryWatcher() {
            _stop = true;
            SetEvent(_overlapped.hEvent); // to wakeup WaitForSingleObject
            _watcher->join();
            CancelIo(_dirHandle);
            CloseHandle(_overlapped.hEvent);
            CloseHandle(_dirHandle);
        }

    private:
        void queueReadChangeRequest() {
            BOOL success = ReadDirectoryChangesW(
                _dirHandle, _buffer.get(), _bufferSizeInBytes, TRUE,
                FILE_NOTIFY_CHANGE_FILE_NAME |
                FILE_NOTIFY_CHANGE_DIR_NAME |
                FILE_NOTIFY_CHANGE_LAST_WRITE,
                NULL, &_overlapped, NULL);
            if (!success) throw std::runtime_error("ReadDirectoryChangesW failed");
        }

        void run() {
            const DWORD waitTimeoutInMs = 100;
            queueReadChangeRequest();
            {
                std::lock_guard<std::mutex> lock(_mutex);
                _running = true;
                _cond.notify_one();
            }
            while (!_stop) {
                DWORD result = WaitForSingleObject(_overlapped.hEvent, INFINITE);
                if (!_stop && result == WAIT_OBJECT_0) {
                    DWORD bytesTransferred;
                    GetOverlappedResult(_dirHandle, &_overlapped, &bytesTransferred, FALSE);
                    if (bytesTransferred == 0) {
                        std::cout << "Overflow" << std::endl;
                    } else {
                        std::size_t offset = 0;
                        PFILE_NOTIFY_INFORMATION info;
                        do
                        {
                            info = reinterpret_cast<PFILE_NOTIFY_INFORMATION>(&_buffer.get()[offset]);
                            offset += info->NextEntryOffset;
                            std::wstring fileNameStr(info->FileName, info->FileNameLength / sizeof(wchar_t));
                            std::filesystem::path fileName(_rootDir / fileNameStr);
                            if (info->Action == FILE_ACTION_ADDED) {
                                std::cout << "Added " << fileName << std::endl;
                            } else if (info->Action == FILE_ACTION_REMOVED) {
                                std::cout << "Removed " << fileName << std::endl;
                            } else if (info->Action == FILE_ACTION_MODIFIED) {
                                std::cout << "Modified " << fileName << std::endl;
                            } else if (info->Action == FILE_ACTION_RENAMED_OLD_NAME) {
                                std::cout << "Renamed old name " << fileName << std::endl;
                            } else if (info->Action == FILE_ACTION_RENAMED_NEW_NAME) {
                                std::cout << "Renamed new name " << fileName << std::endl;
                            }
                        } while (info->NextEntryOffset != 0);
                    }
                    queueReadChangeRequest();
                }
            }
        }
    };

    void iterateDirectory(std::filesystem::path const& dir)
    {
        for (auto const& entry : std::filesystem::directory_iterator(dir)) {
            if (entry.is_directory()) {
                std::filesystem::path s3(dir / entry.path());
                for (auto const& subentry : std::filesystem::directory_iterator(dir)) {
                }
            }
        }
    }
}

int main() {
    std::string tmpDir(std::tmpnam(nullptr));
    std::filesystem::path rootDir(std::string(tmpDir + "_spuriousEvents"));
    std::filesystem::path subDir0(rootDir / "subDir0");
    std::filesystem::path subDir00(subDir0 / "subDir00");
    std::filesystem::create_directory(rootDir);
    std::filesystem::create_directory(subDir0);
    std::filesystem::create_directory(subDir00);
    std::cout << "Created directory " << rootDir.string() << std::endl;
    std::cout << "Created directory " << subDir0.string() << std::endl;
    std::cout << "Created directory " << subDir00.string() << std::endl;

    DirectoryWatcher watcher(rootDir);
    std::cout << std::endl << "Started watching directory tree " << rootDir.string() << std::endl;

    std::cout << std::endl << "Unexpected event during first directory iteration" << std::endl;
    iterateDirectory(subDir0);

    std::this_thread::sleep_for(std::chrono::milliseconds(1000));
    std::cout << std::endl << std::endl;
    std::cout << "No unexpected event during subsequent directory iterations" << std::endl;
    iterateDirectory(subDir0);
    iterateDirectory(subDir0);

    char input = '\0';
    while (input == '\0') std::cin >> input;
}
0

There are 0 best solutions below