如何使用 Inotify 和 libev 在文件夹中获取新添加的文件?
Posted
技术标签:
【中文标题】如何使用 Inotify 和 libev 在文件夹中获取新添加的文件?【英文标题】:How to get new added files in a folder using Inotify along with libev? 【发布时间】:2016-07-21 14:07:28 【问题描述】:我的程序(在 C++ 中)使用 libev 事件循环。我需要在特定文件夹(比如 foo)上查看新文件。
我不能在阻塞模式下使用 Inotify::WaitForEvents(),因为我不想阻塞我的 libev 事件循环。正如inotify documentation 中所建议的那样,我使用 Inotify::SetNonBlock(true) 使其非阻塞。然后将 inotify 文件描述符传递给 libev EV_STAT 以进行监视(如 libev documentation 中所建议的那样)。
当文件夹 foo 中有新文件时,确实会调用 EV_STAT 的 libev 回调。但是,当我使用 Inotify::WaitForEvents() 后跟 Inotify::GetEventCount() 时,我得到零事件。
我怀疑 libev 已经消耗了该事件并将其转换为 EV_STAT 事件。如果是这种情况,我怎样才能获得这些新文件的名称?
我知道 EV_STAT 回调参数中有 inode 编号,但从 inode 编号获取文件名并非易事。所以如果我能得到文件名就更好了。
有什么建议吗?
编辑
我写了一个小程序来重现这个问题。事件似乎并没有丢失。相反,当调用 libev 回调时,inotify 事件还没有到来。当您在新文件中复制时,该事件可能会重新出现。
重现问题的程序:
#include <ev++.h>
#include "inotify-cxx.h"
#include <iostream>
const char * path_to_watch = "/path/to/my/folder";
class ev_inotify_test
InotifyWatch m_watch;
Inotify m_notify;
// for watching new files
ev::stat m_folderWatcher;
public:
ev_inotify_test() : m_watch(path_to_watch, IN_MOVED_TO | IN_CLOSE_WRITE),
m_notify()
void run()
try
start();
// run the loop
ev::get_default_loop().run(0);
catch (InotifyException & e)
std::cout << e.GetMessage() << std::endl;
catch (...)
std::cout << "got an unknown exception." << std::endl;
private:
void start()
m_notify.SetNonBlock(true);
m_notify.Add(m_watch);
m_folderWatcher.set<ev_inotify_test, &ev_inotify_test::cb_stat>(this);
m_folderWatcher.set(path_to_watch);
m_folderWatcher.start();
void cb_stat(ev::stat &w, int revents)
std::cout << "cb_stat called" << std::endl;
try
m_notify.WaitForEvents();
size_t count = m_notify.GetEventCount();
std::cout << "inotify got " << count << " event(s).\n";
while (count > 0)
InotifyEvent event;
bool got_event = m_notify.GetEvent(&event);
std::cout << "inotify confirm got event" << std::endl;
if (got_event)
std::string filename = event.GetName();
std::cout << "test: inotify got file " << filename << std::endl;
--count;
catch (InotifyException &e)
std::cout << "inotify exception occurred: " << e.GetMessage() << std::endl;
catch (...)
std::cout << "Unknown exception in inotify processing occurred!" << std::endl;
;
int main(int argc, char ** argv)
ev_inotify_test().run();
当我复制一个小文件(比如 300 字节)时,会立即检测到该文件。但是如果我复制一个更大的文件(比如 500 kB),直到我复制另一个文件然后我得到两个事件才会有事件。
输出如下:
cb_stat called # test_file_1 (300 bytes) is copied in
inotify got 1 event(s).
inotify confirm got event
test: inotify got file test_file_1
cb_stat called # test_file_2 (500 KB) is copied in
inotify got 0 event(s). # no inotify event
cb_stat called # test_file_3 (300 bytes) is copied in
inotify got 2 event(s).
inotify confirm got event
test: inotify got file test_file_2
inotify confirm got event
test: inotify got file test_file_3
【问题讨论】:
【参考方案1】:我终于弄清楚了问题:我应该使用ev::io来查看inotify的文件描述符,而不是使用ev::stat来查看文件夹。
在示例代码中,m_folderWatcher
的定义应该是:
ev::io m_folderWatcher;
而不是
ev::stat m_folderWatcher;
它应该被初始化为:
m_folderWatcher.set(m_notify.GetDescriptor(), ev::READ);
而不是
m_folderWatcher.set(path_to_watch);
【讨论】:
以上是关于如何使用 Inotify 和 libev 在文件夹中获取新添加的文件?的主要内容,如果未能解决你的问题,请参考以下文章