Tests: Add InodeWatcher and FileWatcher tests

This patch adds some rudimentary tests for InodeWatcher.  It tests the
basic functionality, but maybe there are corner cases I haven't caught.
Additionally, this is our first LibCore test. :^)
This commit is contained in:
sin-ack 2021-05-10 09:27:04 +00:00 committed by Andreas Kling
parent ae9e352104
commit 60eb4adac2
5 changed files with 240 additions and 0 deletions

View file

@ -1,6 +1,7 @@
add_subdirectory(AK)
add_subdirectory(Kernel)
add_subdirectory(LibC)
add_subdirectory(LibCore)
add_subdirectory(LibCompress)
add_subdirectory(LibGfx)
add_subdirectory(LibJS)

View file

@ -5,6 +5,7 @@ set(TEST_SOURCES
${CMAKE_CURRENT_SOURCE_DIR}/TestLibCMkTemp.cpp
${CMAKE_CURRENT_SOURCE_DIR}/TestLibCExec.cpp
${CMAKE_CURRENT_SOURCE_DIR}/TestLibCDirEnt.cpp
${CMAKE_CURRENT_SOURCE_DIR}/TestLibCInodeWatcher.cpp
)
file(GLOB CMD_SOURCES CONFIGURE_DEPENDS "*.cpp")

View file

@ -0,0 +1,170 @@
/*
* Copyright (c) 2021, sin-ack <sin-ack@protonmail.com>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/NumericLimits.h>
#include <Kernel/API/InodeWatcherEvent.h>
#include <Kernel/API/InodeWatcherFlags.h>
#include <LibTest/TestCase.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <utime.h>
u8 buffer[MAXIMUM_EVENT_SIZE];
InodeWatcherEvent* event = reinterpret_cast<InodeWatcherEvent*>(buffer);
static int read_event(int fd)
{
int rc = read(fd, &buffer, MAXIMUM_EVENT_SIZE);
return rc;
}
static String get_event_name()
{
if (event->name_length == 0)
return String();
return String { event->name, event->name_length - 1 };
}
TEST_CASE(inode_watcher_metadata_modified_event)
{
int fd = create_inode_watcher(0);
EXPECT_NE(fd, -1);
int test_fd = creat("/tmp/testfile", 0777);
EXPECT_NE(test_fd, -1);
int wd = inode_watcher_add_watch(fd, "/tmp/testfile", 13, static_cast<unsigned>(InodeWatcherEvent::Type::MetadataModified));
EXPECT_NE(wd, -1);
// "touch" the file
int rc = utime("/tmp/testfile", nullptr);
EXPECT_NE(rc, -1);
rc = read_event(fd);
EXPECT_EQ(event->watch_descriptor, wd);
EXPECT_EQ(event->type, InodeWatcherEvent::Type::MetadataModified);
close(fd);
close(test_fd);
unlink("/tmp/testfile");
}
TEST_CASE(inode_watcher_content_modified_event)
{
int fd = create_inode_watcher(0);
EXPECT_NE(fd, -1);
int test_fd = creat("/tmp/testfile", 0777);
EXPECT_NE(test_fd, -1);
int wd = inode_watcher_add_watch(fd, "/tmp/testfile", 13, static_cast<unsigned>(InodeWatcherEvent::Type::ContentModified));
EXPECT_NE(wd, -1);
int rc = write(test_fd, "test", 4);
EXPECT_NE(rc, -1);
rc = read_event(fd);
EXPECT_NE(rc, -1);
EXPECT_EQ(event->watch_descriptor, wd);
EXPECT_EQ(event->type, InodeWatcherEvent::Type::ContentModified);
close(fd);
close(test_fd);
unlink("/tmp/testfile");
}
TEST_CASE(inode_watcher_deleted_event)
{
int fd = create_inode_watcher(0);
EXPECT_NE(fd, -1);
int test_fd = creat("/tmp/testfile", 0777);
EXPECT_NE(test_fd, -1);
int wd = inode_watcher_add_watch(fd, "/tmp/testfile", 13, static_cast<unsigned>(InodeWatcherEvent::Type::Deleted));
EXPECT_NE(wd, -1);
int rc = unlink("/tmp/testfile");
EXPECT_NE(rc, -1);
rc = read_event(fd);
EXPECT_NE(rc, -1);
EXPECT_EQ(event->watch_descriptor, wd);
EXPECT_EQ(event->type, InodeWatcherEvent::Type::Deleted);
close(fd);
close(test_fd);
}
TEST_CASE(inode_watcher_child_events)
{
int fd = create_inode_watcher(0);
EXPECT_NE(fd, -1);
int wd = inode_watcher_add_watch(fd, "/tmp/", 5, static_cast<unsigned>(InodeWatcherEvent::Type::ChildCreated | InodeWatcherEvent::Type::ChildDeleted));
EXPECT_NE(fd, -1);
int rc = creat("/tmp/testfile", 0777);
EXPECT_NE(rc, -1);
rc = read_event(fd);
EXPECT_NE(rc, -1);
EXPECT_EQ(event->watch_descriptor, wd);
EXPECT_EQ(event->type, InodeWatcherEvent::Type::ChildCreated);
VERIFY(event->name_length > 0);
EXPECT_EQ(get_event_name(), "testfile");
rc = unlink("/tmp/testfile");
EXPECT_NE(rc, -1);
rc = read_event(fd);
EXPECT_NE(rc, -1);
EXPECT_EQ(event->watch_descriptor, wd);
EXPECT_EQ(event->type, InodeWatcherEvent::Type::ChildDeleted);
VERIFY(event->name_length > 0);
EXPECT_EQ(get_event_name(), "testfile");
close(fd);
}
TEST_CASE(inode_watcher_closes_children_on_close)
{
int fd = create_inode_watcher(0);
EXPECT_NE(fd, -1);
int test_fd = creat("/tmp/testfile", 0777);
EXPECT_NE(test_fd, -1);
int wd = inode_watcher_add_watch(fd, "/tmp/testfile", 13, static_cast<unsigned>(InodeWatcherEvent::Type::MetadataModified));
EXPECT_NE(wd, -1);
int rc = utime("/tmp/testfile", nullptr);
EXPECT_NE(rc, -1);
close(fd);
rc = read_event(fd);
EXPECT_EQ(rc, -1);
EXPECT_EQ(errno, EBADF);
close(test_fd);
unlink("/tmp/testfile");
}
TEST_CASE(inode_watcher_nonblock)
{
int fd = create_inode_watcher(static_cast<unsigned>(InodeWatcherFlags::Nonblock));
EXPECT_NE(fd, -1);
int rc = read_event(fd);
EXPECT_EQ(rc, -1);
EXPECT_EQ(errno, EAGAIN);
close(fd);
}

View file

@ -0,0 +1,8 @@
set(
TEST_SOURCES
${CMAKE_CURRENT_SOURCE_DIR}/TestLibCoreFileWatcher.cpp
)
foreach(source ${TEST_SOURCES})
serenity_test(${source} LibCore)
endforeach()

View file

@ -0,0 +1,60 @@
/*
* Copyright (c) 2021, the SerenityOS developers.
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <Kernel/API/InodeWatcherEvent.h>
#include <LibCore/EventLoop.h>
#include <LibCore/FileWatcher.h>
#include <LibCore/Timer.h>
#include <LibTest/TestCase.h>
#include <fcntl.h>
#include <unistd.h>
TEST_CASE(file_watcher_child_events)
{
auto event_loop = Core::EventLoop();
auto maybe_file_watcher = Core::FileWatcher::create();
EXPECT_NE(maybe_file_watcher.is_error(), true);
auto file_watcher = maybe_file_watcher.release_value();
auto watch_result = file_watcher->add_watch("/tmp/",
Core::FileWatcherEvent::Type::ChildCreated
| Core::FileWatcherEvent::Type::ChildDeleted);
EXPECT_NE(watch_result.is_error(), true);
int event_count = 0;
file_watcher->on_change = [&](Core::FileWatcherEvent const& event) {
if (event_count == 0) {
EXPECT_EQ(event.event_path, "/tmp/testfile");
EXPECT_EQ(event.type, Core::FileWatcherEvent::Type::ChildCreated);
} else if (event_count == 1) {
EXPECT_EQ(event.event_path, "/tmp/testfile");
EXPECT_EQ(event.type, Core::FileWatcherEvent::Type::ChildDeleted);
event_loop.quit(0);
}
event_count++;
};
auto timer1 = Core::Timer::create_single_shot(500, [&] {
int rc = creat("/tmp/testfile", 0777);
EXPECT_NE(rc, -1);
});
timer1->start();
auto timer2 = Core::Timer::create_single_shot(1000, [&] {
int rc = unlink("/tmp/testfile");
EXPECT_NE(rc, -1);
});
timer2->start();
auto catchall_timer = Core::Timer::create_single_shot(2000, [&] {
VERIFY_NOT_REACHED();
});
catchall_timer->start();
event_loop.exec();
}