chromium/content/browser/file_system_access/file_path_watcher/file_path_watcher_unittest.cc

// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "content/browser/file_system_access/file_path_watcher/file_path_watcher.h"

#include <list>
#include <memory>
#include <string>
#include <vector>

#include "base/atomic_sequence_num.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/files/scoped_temp_dir.h"
#include "base/functional/bind.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/run_loop.h"
#include "base/sequence_checker.h"
#include "base/strings/stringprintf.h"
#include "base/task/bind_post_task.h"
#include "base/task/sequenced_task_runner.h"
#include "base/task/single_thread_task_runner.h"
#include "base/test/bind.h"
#include "base/test/run_until.h"
#include "base/test/task_environment.h"
#include "base/test/test_file_util.h"
#include "base/test/test_timeouts.h"
#include "base/thread_annotations.h"
#include "base/threading/thread.h"
#include "build/build_config.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"

#if BUILDFLAG(IS_WIN)
#include <windows.h>

#include <aclapi.h>
#elif BUILDFLAG(IS_POSIX)
#include <sys/stat.h>
#endif

#if BUILDFLAG(IS_ANDROID)
#include "base/android/path_utils.h"
#endif  // BUILDFLAG(IS_ANDROID)

#if BUILDFLAG(IS_POSIX)
#include "base/files/file_descriptor_watcher_posix.h"
#endif  // BUILDFLAG(IS_POSIX)

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
#include "base/format_macros.h"
#include "content/browser/file_system_access/file_path_watcher/file_path_watcher_inotify.h"
#endif

namespace content {

namespace {

base::AtomicSequenceNumber g_next_delegate_id;

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) || \
    BUILDFLAG(IS_WIN)
// inotify fires two events - one for each file creation + modification.
constexpr size_t kExpectedEventsForNewFileWrite =;
#else
constexpr size_t kExpectedEventsForNewFileWrite = 1;
#endif

enum class ExpectedEventsSinceLastWait {};

struct Event {};
EventListMatcher;

Event ToEvent(const FilePathWatcher::ChangeInfo& change_info,
              const base::FilePath& path,
              bool error) {}

std::ostream& operator<<(std::ostream& os,
                         const FilePathWatcher::ChangeType& change_type) {}

std::ostream& operator<<(std::ostream& os,
                         const FilePathWatcher::FilePathType& file_path_type) {}

std::ostream& operator<<(std::ostream& os,
                         const FilePathWatcher::ChangeInfo& change_info) {}

std::ostream& operator<<(std::ostream& os, const Event& event) {}

void SpinEventLoopForABit() {}

// Returns the reason why `value` matches, or doesn't match, `matcher`.
template <typename MatcherType, typename Value>
std::string Explain(const MatcherType& matcher, const Value& value) {}

#if !BUILDFLAG(IS_MAC)
inline constexpr auto HasPath =;
inline constexpr auto HasErrored =;
inline constexpr auto HasModifiedPath =;
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) || \
    BUILDFLAG(IS_WIN)
inline constexpr auto HasMovedFromPath =;
#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
        // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_WIN)
inline constexpr auto HasNoMovedFromPath =;
inline constexpr auto IsType =;

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) || \
    BUILDFLAG(IS_WIN)
inline constexpr auto IsFile =;
inline constexpr auto IsDirectory =;
#endif

#if !BUILDFLAG(IS_LINUX) && !BUILDFLAG(IS_CHROMEOS) && !BUILDFLAG(IS_ANDROID)
inline constexpr auto IsUnknownPathType = []() {
  return testing::Field(
      &Event::change_info,
      testing::Field(&FilePathWatcher::ChangeInfo::file_path_type,
                     FilePathWatcher::FilePathType::kUnknown));
};
#endif
#endif  // !BUILDFLAG(IS_MAC)

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)
inline constexpr auto IsDeletedFile =;
inline constexpr auto IsDeletedDirectory =;

// TODO(crbug.com/341372596): A file move is reported as a directory on linux.
inline constexpr auto IsMovedFile =;
inline constexpr auto ModifiedMatcher =;

#elif BUILDFLAG(IS_WIN)
// Windows figures out if a file path is a directory or file with `GetFileInfo`,
// but since the file is deleted, it can't know.
//
// This also needs to be used for events for a deleted file before it's actually
// deleted since the file path type still can't be determined.
inline constexpr auto IsDeletedFile = []() {
  return testing::AnyOf(IsFile(), IsUnknownPathType());
};
inline constexpr auto IsDeletedDirectory = []() {
  return testing::AnyOf(IsDirectory(), IsUnknownPathType());
};

inline constexpr auto IsMovedFile = IsFile;

// WriteFile causes two writes on Windows because it calls two syscalls:
// ::CreateFile and ::WriteFile.
inline constexpr auto ModifiedMatcher = [](base::FilePath reported_path,
                                           base::FilePath modified_path) {
  const auto modified_matcher =
      testing::AllOf(HasPath(reported_path), testing::Not(HasErrored()),
                     IsFile(), IsType(FilePathWatcher::ChangeType::kModified),
                     HasModifiedPath(modified_path), HasNoMovedFromPath());
  return testing::ElementsAreArray({modified_matcher, modified_matcher});
};
#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
        // BUILDFLAG(IS_ANDROID)

// Enables an accumulative, add-as-you-go pattern for expecting events:
//   - Do something that should fire `event1` on `delegate`
//   - Add `event1` to an `accumulated_event_expecter`
//   - Wait until `delegate` matches { `event1` }
//   - Do something that should fire `event2` on `delegate`
//   - Add `event2` to an `accumulated_event_expecter`
//   - Wait until `delegate` matches { `event1`, `event2` }
//   - ...
//
// These tests use an accumulative pattern due to the potential for
// false-positives, given that all we know is the number of changes at a given
// path (which is often fixed) and whether or not an error occurred (which is
// rare).
//
// TODO(crbug.com/40260973): This is not a common pattern. Generally,
// expectations are specified all-in-one at the start of a test, like so:
//   - Expect events { `event1`, `event2` }
//   - Do something that should fire `event1` on `delegate`
//   - Do something that should fire `event2` on `delegate`
//   - Wait until `delegate` matches { `event1`, `event2` }
//
// The potential for false-positives is much less if event types are known. We
// should consider moving towards the latter pattern
// (see `FilePathWatcherWithChangeInfoTest`) once that is supported.
class AccumulatingEventExpecter {};

class TestDelegateBase {};

// Receives and accumulates notifications from a specific `FilePathWatcher`.
// This class is not thread safe. All methods must be called from the sequence
// the instance is constructed on.
class TestDelegate final : public TestDelegateBase {};

}  // namespace

#if BUILDFLAG(IS_FUCHSIA)
// FilePatchWatcherImpl is not implemented (see crbug.com/851641).
// Disable all tests.
#define FilePathWatcherTest
#endif

class FilePathWatcherTest : public testing::Test {};

bool FilePathWatcherTest::SetupWatch(const base::FilePath& target,
                                     FilePathWatcher* watcher,
                                     TestDelegateBase* delegate,
                                     FilePathWatcher::Type watch_type) {}

bool FilePathWatcherTest::SetupWatchWithOptions(
    const base::FilePath& target,
    FilePathWatcher* watcher,
    TestDelegateBase* delegate,
    FilePathWatcher::WatchOptions watch_options) {}

bool FilePathWatcherTest::SetupWatchWithChangeInfo(
    const base::FilePath& target,
    FilePathWatcher* watcher,
    TestDelegateBase* delegate,
    FilePathWatcher::WatchOptions watch_options) {}

// Basic test: Create the file and verify that we notice.
TEST_F(FilePathWatcherTest, NewFile) {}

// Basic test: Create the directory and verify that we notice.
TEST_F(FilePathWatcherTest, NewDirectory) {}

// Basic test: Create the directory and verify that we notice.
TEST_F(FilePathWatcherTest, NewDirectoryRecursiveWatch) {}

// Verify that modifying the file is caught.
TEST_F(FilePathWatcherTest, ModifiedFile) {}

// Verify that creating the parent directory of the watched file is not caught.
TEST_F(FilePathWatcherTest, CreateParentDirectory) {}

// Verify that changes to the sibling of the watched file are not caught.
TEST_F(FilePathWatcherTest, CreateSiblingFile) {}

// Verify that changes to the sibling of the parent of the watched file are not
// caught.
TEST_F(FilePathWatcherTest, CreateParentSiblingFile) {}

// Verify that moving an unwatched file to a watched path is caught.
TEST_F(FilePathWatcherTest, MovedToFile) {}

// Verify that moving the watched file to an unwatched path is caught.
TEST_F(FilePathWatcherTest, MovedFromFile) {}

TEST_F(FilePathWatcherTest, DeletedFile) {}

#if BUILDFLAG(IS_WIN)
TEST_F(FilePathWatcherTest, WindowsBufferOverflow) {
  FilePathWatcher watcher;
  TestDelegate delegate;
  AccumulatingEventExpecter event_expecter;
  ASSERT_TRUE(SetupWatch(test_file(), &watcher, &delegate,
                         FilePathWatcher::Type::kNonRecursive));

  {
    // Block the Watch thread.
    base::AutoLock auto_lock(watcher.GetWatchThreadLockForTest());

    // Generate an event that will try to acquire the lock on the watch thread.
    ASSERT_TRUE(WriteFile(test_file(), "content"));

    // The packet size plus the path size. `WriteFile` generates two events so
    // it's twice that.
    const size_t kWriteFileEventSize =
        (sizeof(FILE_NOTIFY_INFORMATION) + test_file().AsUTF8Unsafe().size()) *
        2;

    // The max size that's allowed for network drives:
    // https://learn.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-readdirectorychangesw#remarks.
    const size_t kMaxBufferSize = 64 * 1024;

    for (size_t bytes_in_buffer = 0; bytes_in_buffer < kMaxBufferSize;
         bytes_in_buffer += kWriteFileEventSize) {
      WriteFile(test_file(), "content");
    }
  }

  // The initial `WriteFile` generates an event.
  event_expecter.AddExpectedEventForPath(test_file());
  // The rest should only appear as a buffer overflow.
  event_expecter.AddExpectedEventForPath(test_file());
  delegate.RunUntilEventsMatch(event_expecter);
}
#endif

namespace {

// Used by the DeleteDuringNotify test below.
// Deletes the FilePathWatcher when it's notified.
class Deleter final : public TestDelegateBase {};

}  // namespace

// Verify that deleting a watcher during the callback doesn't crash.
TEST_F(FilePathWatcherTest, DeleteDuringNotify) {}

// Verify that deleting the watcher works even if there is a pending
// notification.
TEST_F(FilePathWatcherTest, DestroyWithPendingNotification) {}

TEST_F(FilePathWatcherTest, MultipleWatchersSingleFile) {}

// Verify that watching a file whose parent directory doesn't exist yet works if
// the directory and file are created eventually.
TEST_F(FilePathWatcherTest, NonExistentDirectory) {}

// Exercises watch reconfiguration for the case that directories on the path
// are rapidly created.
TEST_F(FilePathWatcherTest, DirectoryChain) {}

// Windows doesn't allow the target directory to be deleted while there is a
// FilePathWatcher watching it.
#if !BUILDFLAG(IS_WIN)
TEST_F(FilePathWatcherTest, DisappearingDirectory) {}
#endif

// Tests that a file that is deleted and reappears is tracked correctly.
TEST_F(FilePathWatcherTest, DeleteAndRecreate) {}

// TODO(crbug.com/40263777): Split into smaller tests.
TEST_F(FilePathWatcherTest, WatchDirectory) {}

TEST_F(FilePathWatcherTest, MoveParent) {}

TEST_F(FilePathWatcherTest, RecursiveWatch) {}

#if BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_MAC)
// Apps cannot create symlinks on Android in /sdcard as /sdcard uses the
// "fuse" file system, while /data uses "ext4".  Running these tests in /data
// would be preferable and allow testing file attributes and symlinks.
// TODO(pauljensen): Re-enable when crbug.com/475568 is fixed and SetUp() places
// the |temp_dir_| in /data.
//
// This test is disabled on Fuchsia since it doesn't support symlinking.
//
// This test is disabled on Mac since recursive watches aren't supported for
// symlinks.
TEST_F(FilePathWatcherTest, RecursiveWithSymLink) {}
#endif  // BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_MAC)

TEST_F(FilePathWatcherTest, MoveChild) {}

TEST_F(FilePathWatcherTest, MoveOverwritingFile) {}

// Verify that changing attributes on a file is caught
#if BUILDFLAG(IS_ANDROID)
// Apps cannot change file attributes on Android in /sdcard as /sdcard uses the
// "fuse" file system, while /data uses "ext4".  Running these tests in /data
// would be preferable and allow testing file attributes and symlinks.
// TODO(pauljensen): Re-enable when crbug.com/475568 is fixed and SetUp() places
// the |temp_dir_| in /data.
#define FileAttributesChanged
#endif  // BUILDFLAG(IS_ANDROID)

// This test is disabled on Mac because we don't support reporting file metadata
// changes on FSEvents.
#if !BUILDFLAG(IS_MAC)
TEST_F(FilePathWatcherTest, NoEventWhenFileAttributesChanged) {}
#endif  // !BUILDFLAG(IS_MAC)

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)

// Verify that creating a symlink is caught.
TEST_F(FilePathWatcherTest, CreateLink) {}

// Verify that deleting a symlink is caught.
TEST_F(FilePathWatcherTest, DeleteLink) {}

// Verify that modifying a target file that a link is pointing to
// when we are watching the link is caught.
TEST_F(FilePathWatcherTest, ModifiedLinkedFile) {}

// Verify that creating a target file that a link is pointing to
// when we are watching the link is caught.
TEST_F(FilePathWatcherTest, CreateTargetLinkedFile) {}

// Verify that deleting a target file that a link is pointing to
// when we are watching the link is caught.
TEST_F(FilePathWatcherTest, DeleteTargetLinkedFile) {}

// Verify that watching a file whose parent directory is a link that
// doesn't exist yet works if the symlink is created eventually.
TEST_F(FilePathWatcherTest, LinkedDirectoryPart1) {}

// Verify that watching a file whose parent directory is a
// dangling symlink works if the directory is created eventually.
// TODO(crbug.com/40263777): Add test coverage for symlinked file
// creation independent of a corresponding write.
TEST_F(FilePathWatcherTest, LinkedDirectoryPart2) {}

// Verify that watching a file with a symlink on the path
// to the file works.
TEST_F(FilePathWatcherTest, LinkedDirectoryPart3) {}

// Regression tests that FilePathWatcherImpl does not leave its reference in
// `g_inotify_reader` due to a race in recursive watch.
// See https://crbug.com/990004.
TEST_F(FilePathWatcherTest, RacyRecursiveWatch) {}

// Verify that "Watch()" returns false and callback is not invoked when limit is
// hit during setup.
TEST_F(FilePathWatcherTest, InotifyLimitInWatch) {}

// Verify that "error=true" callback happens when limit is hit during update.
TEST_F(FilePathWatcherTest, InotifyLimitInUpdate) {}

// Similar to InotifyLimitInUpdate but test a recursive watcher.
TEST_F(FilePathWatcherTest, InotifyLimitInUpdateRecursive) {}

#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)

// TODO(fxbug.dev/60109): enable BUILDFLAG(IS_FUCHSIA) when implemented.
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID)

TEST_F(FilePathWatcherTest, ReturnFullPath_RecursiveInRootFolder) {}

TEST_F(FilePathWatcherTest, ReturnFullPath_RecursiveInNestedFolder) {}

TEST_F(FilePathWatcherTest, ReturnFullPath_NonRecursiveInRootFolder) {}

TEST_F(FilePathWatcherTest, ReturnFullPath_NonRecursiveRemoveEnclosingFolder) {}

TEST_F(FilePathWatcherTest, ReturnWatchedPath_RecursiveInRootFolder) {}

TEST_F(FilePathWatcherTest, ReturnWatchedPath_NonRecursiveInRootFolder) {}

#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
        // BUILDFLAG(IS_ANDROID)

namespace {

enum Permission {};

#if BUILDFLAG(IS_APPLE)
bool ChangeFilePermissions(const base::FilePath& path,
                           Permission perm,
                           bool allow) {
  struct stat stat_buf;

  if (stat(path.value().c_str(), &stat_buf) != 0) {
    return false;
  }

  mode_t mode = 0;
  switch (perm) {
    case Read:
      mode = S_IRUSR | S_IRGRP | S_IROTH;
      break;
    case Write:
      mode = S_IWUSR | S_IWGRP | S_IWOTH;
      break;
    case Execute:
      mode = S_IXUSR | S_IXGRP | S_IXOTH;
      break;
    default:
      ADD_FAILURE() << "unknown perm " << perm;
      return false;
  }
  if (allow) {
    stat_buf.st_mode |= mode;
  } else {
    stat_buf.st_mode &= ~mode;
  }
  return chmod(path.value().c_str(), stat_buf.st_mode) == 0;
}
#endif  // BUILDFLAG(IS_APPLE)

}  // namespace

#if BUILDFLAG(IS_APPLE)
// Linux implementation of FilePathWatcher doesn't catch attribute changes.
// http://crbug.com/78043
// Windows implementation of FilePathWatcher catches attribute changes that
// don't affect the path being watched.
// http://crbug.com/78045

// Verify that changing attributes on a directory works.
TEST_F(FilePathWatcherTest, DirAttributesChanged) {
  base::FilePath test_dir1(
      temp_dir_.GetPath().AppendASCII("DirAttributesChangedDir1"));
  base::FilePath test_dir2(test_dir1.AppendASCII("DirAttributesChangedDir2"));
  base::FilePath test_file(test_dir2.AppendASCII("DirAttributesChangedFile"));
  // Setup a directory hierarchy.
  ASSERT_TRUE(CreateDirectory(test_dir1));
  ASSERT_TRUE(CreateDirectory(test_dir2));
  ASSERT_TRUE(WriteFile(test_file, "content"));

  FilePathWatcher watcher;
  TestDelegate delegate;
  AccumulatingEventExpecter event_expecter;
  ASSERT_TRUE(SetupWatch(test_file, &watcher, &delegate,
                         FilePathWatcher::Type::kNonRecursive));

  // We should not get notified in this case as it hasn't affected our ability
  // to access the file.
  ASSERT_TRUE(ChangeFilePermissions(test_dir1, Read, false));
  ASSERT_TRUE(ChangeFilePermissions(test_dir1, Read, true));
  // TODO(crbug.com/40263777): Expect that no events are fired.

  // We should get notified in this case because filepathwatcher can no
  // longer access the file.
  ASSERT_TRUE(ChangeFilePermissions(test_dir1, Execute, false));
  event_expecter.AddExpectedEventForPath(test_file);
  delegate.RunUntilEventsMatch(event_expecter);

  ASSERT_TRUE(ChangeFilePermissions(test_dir1, Execute, true));
  // TODO(crbug.com/40263777): Expect that no events are fired.
}

#endif  // BUILDFLAG(IS_APPLE)

#if BUILDFLAG(IS_APPLE)

// Fail fast if trying to trivially watch a non-existent item.
TEST_F(FilePathWatcherTest, TrivialNoDir) {
  const base::FilePath tmp_dir = temp_dir_.GetPath();
  const base::FilePath non_existent = tmp_dir.Append(FILE_PATH_LITERAL("nope"));

  FilePathWatcher watcher;
  TestDelegate delegate;
  ASSERT_FALSE(SetupWatch(non_existent, &watcher, &delegate,
                          FilePathWatcher::Type::kTrivial));
}

// Succeed starting a watch on a directory.
TEST_F(FilePathWatcherTest, TrivialDirStart) {
  const base::FilePath tmp_dir = temp_dir_.GetPath();

  FilePathWatcher watcher;
  TestDelegate delegate;
  ASSERT_TRUE(SetupWatch(tmp_dir, &watcher, &delegate,
                         FilePathWatcher::Type::kTrivial));
}

// Observe a change on a directory
TEST_F(FilePathWatcherTest, TrivialDirChange) {
  const base::FilePath tmp_dir = temp_dir_.GetPath();

  FilePathWatcher watcher;
  TestDelegate delegate;
  AccumulatingEventExpecter event_expecter;
  ASSERT_TRUE(SetupWatch(tmp_dir, &watcher, &delegate,
                         FilePathWatcher::Type::kTrivial));

  ASSERT_TRUE(TouchFile(tmp_dir, base::Time::Now(), base::Time::Now()));
  event_expecter.AddExpectedEventForPath(tmp_dir);
  delegate.RunUntilEventsMatch(event_expecter);
}

// Observe no change when a parent is modified.
TEST_F(FilePathWatcherTest, TrivialParentDirChange) {
  const base::FilePath tmp_dir = temp_dir_.GetPath();
  const base::FilePath sub_dir1 = tmp_dir.Append(FILE_PATH_LITERAL("subdir"));
  const base::FilePath sub_dir2 =
      sub_dir1.Append(FILE_PATH_LITERAL("subdir_redux"));

  ASSERT_TRUE(CreateDirectory(sub_dir1));
  ASSERT_TRUE(CreateDirectory(sub_dir2));

  FilePathWatcher watcher;
  TestDelegate delegate;
  AccumulatingEventExpecter event_expecter;
  ASSERT_TRUE(SetupWatch(sub_dir2, &watcher, &delegate,
                         FilePathWatcher::Type::kTrivial));

  // There should be no notification for a change to |sub_dir2|'s parent.
  ASSERT_TRUE(Move(sub_dir1, tmp_dir.Append(FILE_PATH_LITERAL("over_here"))));
  delegate.RunUntilEventsMatch(event_expecter);
}

// Do not crash when a directory is moved; https://crbug.com/1156603.
TEST_F(FilePathWatcherTest, TrivialDirMove) {
  const base::FilePath tmp_dir = temp_dir_.GetPath();
  const base::FilePath sub_dir = tmp_dir.Append(FILE_PATH_LITERAL("subdir"));

  ASSERT_TRUE(CreateDirectory(sub_dir));

  FilePathWatcher watcher;
  TestDelegate delegate;
  AccumulatingEventExpecter event_expecter;
  ASSERT_TRUE(SetupWatch(sub_dir, &watcher, &delegate,
                         FilePathWatcher::Type::kTrivial));

  ASSERT_TRUE(Move(sub_dir, tmp_dir.Append(FILE_PATH_LITERAL("over_here"))));
  event_expecter.AddExpectedEventForPath(sub_dir, /**error=*/true);
  delegate.RunUntilEventsMatch(event_expecter);
}

#endif  // BUILDFLAG(IS_APPLE)

// TODO(b/359174510): Disabled on Mac due to flakiness. When change info is
// reported, FSEvents sometimes does not report changes on time, before the test
// timer times out. Re-enable this test class on Mac once the flakes are
// resolved.
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) || \
    BUILDFLAG(IS_WIN)
// TODO(crbug.com/40263777): Ideally most all of the tests above would be
// parameterized in this way.
class FilePathWatcherWithChangeInfoTest
    : public FilePathWatcherTest,
      public testing::WithParamInterface<
          std::tuple<FilePathWatcher::Type, bool>> {};

TEST_P(FilePathWatcherWithChangeInfoTest, NewFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, NewDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, ModifiedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, MovedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DeletedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DeletedDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, MultipleWatchersSingleFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, NonExistentDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DirectoryChain) {}

// Windows doesn't allow the target directory to be deleted while there is a
// FilePathWatcher watching it.
#if !BUILDFLAG(IS_WIN)
TEST_P(FilePathWatcherWithChangeInfoTest, DisappearingDirectory) {}
#endif

TEST_P(FilePathWatcherWithChangeInfoTest, DeleteAndRecreate) {}

TEST_P(FilePathWatcherWithChangeInfoTest, WatchDirectory) {}

// TODO(crbug.com/362715979): This test is disabled on Mac due to unexpected,
// test-specific behavior - we receive no FSEvents events when the ancestor dir
// of the watched target is moved out-of-scope. Since we never receive events
// from FSEvents, it's impossible for us to report the expected 'delete' event.
// This behavior does not repro in manual testing. In manual tests of this use
// case, we receive all events as expected, including the equivalent 'delete'
// event that never arrives in the unittest.
#if !BUILDFLAG(IS_MAC)
TEST_P(FilePathWatcherWithChangeInfoTest, MoveParent) {}
#endif  // !BUILDFLAG(IS_MAC)

TEST_P(FilePathWatcherWithChangeInfoTest, MoveChild) {}

TEST_P(FilePathWatcherWithChangeInfoTest, MoveChildWithinWatchedScope) {}

TEST_P(FilePathWatcherWithChangeInfoTest, MoveChildOutOrIntoWatchedScope) {}

// TODO(pauljensen): Re-enable when crbug.com/475568 is fixed and SetUp() places
// the |temp_dir_| in /data.
#if !BUILDFLAG(IS_ANDROID)

// This test is disabled on Mac because we don't support reporting file metadata
// changes on FSEvents.
#if !BUILDFLAG(IS_MAC)
TEST_P(FilePathWatcherWithChangeInfoTest, NoEventWhenFileAttributesChanged) {}
#endif  // !BUILDFLAG(IS_MAC)

#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
TEST_P(FilePathWatcherWithChangeInfoTest, CreateLink) {}

// Unfortunately this test case only works if the link target exists.
// TODO(craig) fix this as part of crbug.com/91561.
TEST_P(FilePathWatcherWithChangeInfoTest, DeleteLink) {}

TEST_P(FilePathWatcherWithChangeInfoTest, ModifiedLinkedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, CreateTargetLinkedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DeleteTargetLinkedFile) {}

TEST_P(FilePathWatcherWithChangeInfoTest, LinkedDirectoryPart1) {}

TEST_P(FilePathWatcherWithChangeInfoTest, LinkedDirectoryPart2) {}

TEST_P(FilePathWatcherWithChangeInfoTest, LinkedDirectoryPart3) {}
#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
#endif  // !BUILDFLAG(IS_ANDROID)

TEST_P(FilePathWatcherWithChangeInfoTest, CreatedFileInDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, ModifiedFileInDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DeletedFileInDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, FileInDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, DirectoryInDirectory) {}

TEST_P(FilePathWatcherWithChangeInfoTest, NestedDirectoryInDirectory) {}

// Windows doesn't allow the target directory to be deleted while there is a
// FilePathWatcher watching it.
#if !BUILDFLAG(IS_WIN)
TEST_P(FilePathWatcherWithChangeInfoTest, DeleteDirectoryRecursively) {}
#endif

INSTANTIATE_TEST_SUITE_P();

#else

#if !BUILDFLAG(IS_MAC)
TEST_F(FilePathWatcherTest, UseDummyChangeInfoIfNotSupported) {
  const auto matcher = testing::ElementsAre(testing::AllOf(
      HasPath(test_file()), testing::Not(HasErrored()), IsUnknownPathType(),
      IsType(FilePathWatcher::ChangeType::kUnknown),
      HasModifiedPath(base::FilePath()), HasNoMovedFromPath()));

  FilePathWatcher watcher;
  TestDelegate delegate;
  ASSERT_TRUE(
      SetupWatchWithChangeInfo(test_file(), &watcher, &delegate,
                               {.type = FilePathWatcher::Type::kNonRecursive}));

  ASSERT_TRUE(CreateDirectory(test_file()));
  delegate.RunUntilEventsMatch(matcher);
}
#endif  // !BUILDFLAG(IS_MAC)

#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) ||
        // BUILDFLAG(IS_ANDROID) || BUILDFLAG(IS_WIN)

}  // namespace content