mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-22 07:42:02 +00:00
Implement FileMonitor as a wrapper for QFileSystemWatcher
The new file monitor supports both boost and qt signals. It is implemented in a ressource-safe way.
This commit is contained in:
parent
22edb3df96
commit
caa54e80ee
@ -15,6 +15,9 @@
|
|||||||
#ifndef FILEMONITOR_H
|
#ifndef FILEMONITOR_H
|
||||||
#define FILEMONITOR_H
|
#define FILEMONITOR_H
|
||||||
|
|
||||||
|
// TODO: Remove FileMonitor
|
||||||
|
#include "support/FileMonitor2.h"
|
||||||
|
|
||||||
#include <boost/signals2.hpp>
|
#include <boost/signals2.hpp>
|
||||||
|
|
||||||
namespace lyx {
|
namespace lyx {
|
||||||
@ -28,7 +31,8 @@ public:
|
|||||||
/** Once monitoring begins, the file will be monitored every
|
/** Once monitoring begins, the file will be monitored every
|
||||||
* interval ms.
|
* interval ms.
|
||||||
*
|
*
|
||||||
* FIXME: rewrite and simplify using an encapsulation of QFileSystemWatcher.
|
* This is now obsoleted by FileMonitor2 based on QFileSystemWatcher.
|
||||||
|
* FIXME: Remove FileMonitor
|
||||||
*/
|
*/
|
||||||
FileMonitor(FileName const & file_with_path, int interval);
|
FileMonitor(FileName const & file_with_path, int interval);
|
||||||
|
|
||||||
|
211
src/support/FileMonitor2.cpp
Normal file
211
src/support/FileMonitor2.cpp
Normal file
@ -0,0 +1,211 @@
|
|||||||
|
/**
|
||||||
|
* \file FileMonitor.cpp
|
||||||
|
* This file is part of LyX, the document processor.
|
||||||
|
* Licence details can be found in the file COPYING.
|
||||||
|
*
|
||||||
|
* \author Guillaume Munch
|
||||||
|
*
|
||||||
|
* Full author contact details are available in file CREDITS.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include "support/FileMonitor2.h"
|
||||||
|
|
||||||
|
#include "support/debug.h"
|
||||||
|
#include "support/FileName.h"
|
||||||
|
#include "support/qstring_helpers.h"
|
||||||
|
#include "support/unique_ptr.h"
|
||||||
|
|
||||||
|
#include <QFile>
|
||||||
|
#include <QSignalBlocker>
|
||||||
|
#include <QTimer>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
namespace lyx {
|
||||||
|
namespace support {
|
||||||
|
|
||||||
|
|
||||||
|
FileSystemWatcher & FileSystemWatcher::instance()
|
||||||
|
{
|
||||||
|
// This thread-safe because QFileSystemWatcher is thread-safe.
|
||||||
|
static FileSystemWatcher f;
|
||||||
|
return f;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileSystemWatcher::FileSystemWatcher()
|
||||||
|
: qwatcher_(make_unique<QFileSystemWatcher>())
|
||||||
|
{}
|
||||||
|
|
||||||
|
|
||||||
|
//static
|
||||||
|
FileMonitorPtr FileSystemWatcher::monitor(FileName const & file_with_path)
|
||||||
|
{
|
||||||
|
FileSystemWatcher & f = instance();
|
||||||
|
string const filename = file_with_path.absFileName();
|
||||||
|
weak_ptr<FileMonitorGuard> & wptr = f.store_[filename];
|
||||||
|
if (shared_ptr<FileMonitorGuard> mon = wptr.lock())
|
||||||
|
return make_unique<FileMonitor2>(mon);
|
||||||
|
auto mon = make_shared<FileMonitorGuard>(filename, f.qwatcher_.get());
|
||||||
|
wptr = mon;
|
||||||
|
return make_unique<FileMonitor2>(mon);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//static
|
||||||
|
void FileSystemWatcher::debug()
|
||||||
|
{
|
||||||
|
FileSystemWatcher & f = instance();
|
||||||
|
QStringList q_files = f.qwatcher_->files();
|
||||||
|
for (pair<string, weak_ptr<FileMonitorGuard>> pair : f.store_) {
|
||||||
|
string const & name = pair.first;
|
||||||
|
if (!pair.second.expired()) {
|
||||||
|
if (!q_files.contains(toqstr(name)))
|
||||||
|
LYXERR0("Monitored but not QFileSystemWatched (bad): " << name);
|
||||||
|
else {
|
||||||
|
//LYXERR0("Monitored and QFileSystemWatched (good): " << name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (QString const & qname : q_files) {
|
||||||
|
string const name = fromqstr(qname);
|
||||||
|
weak_ptr<FileMonitorGuard> & wptr = f.store_[name];
|
||||||
|
if (wptr.expired())
|
||||||
|
LYXERR0("QFileSystemWatched but not monitored (bad): " << name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitorGuard::FileMonitorGuard(string const & filename,
|
||||||
|
QFileSystemWatcher * qwatcher)
|
||||||
|
: filename_(filename), qwatcher_(qwatcher)
|
||||||
|
{
|
||||||
|
QObject::connect(qwatcher, SIGNAL(fileChanged(QString const &)),
|
||||||
|
this, SLOT(notifyChange(QString const &)));
|
||||||
|
if (qwatcher_->files().contains(toqstr(filename)))
|
||||||
|
LYXERR0("This file is already being QFileSystemWatched: " << filename
|
||||||
|
<< ". This should not happen.");
|
||||||
|
refresh();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitorGuard::~FileMonitorGuard()
|
||||||
|
{
|
||||||
|
qwatcher_->removePath(toqstr(filename_));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitorGuard::refresh(bool new_file)
|
||||||
|
{
|
||||||
|
QString const qfilename = toqstr(filename_);
|
||||||
|
if(!qwatcher_->files().contains(qfilename)) {
|
||||||
|
bool exists = QFile(qfilename).exists();
|
||||||
|
if (!exists || !qwatcher_->addPath(qfilename)) {
|
||||||
|
if (exists)
|
||||||
|
LYXERR(Debug::FILES,
|
||||||
|
"Could not add path to QFileSystemWatcher: "
|
||||||
|
<< filename_);
|
||||||
|
QTimer::singleShot(1000, this, [=](){
|
||||||
|
refresh(new_file || !exists);
|
||||||
|
});
|
||||||
|
} else if (exists && new_file)
|
||||||
|
Q_EMIT fileChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitorGuard::notifyChange(QString const & path)
|
||||||
|
{
|
||||||
|
if (path == toqstr(filename_)) {
|
||||||
|
Q_EMIT fileChanged();
|
||||||
|
// If the file has been modified by delete-move, we are notified of the
|
||||||
|
// deletion but we no longer track the file. See
|
||||||
|
// <https://bugreports.qt.io/browse/QTBUG-46483> (not a bug).
|
||||||
|
refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitor2::FileMonitor2(std::shared_ptr<FileMonitorGuard> monitor)
|
||||||
|
: monitor_(monitor)
|
||||||
|
{
|
||||||
|
connectToFileMonitorGuard();
|
||||||
|
refresh();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitor2::connectToFileMonitorGuard()
|
||||||
|
{
|
||||||
|
QObject::connect(monitor_.get(), SIGNAL(fileChanged()),
|
||||||
|
this, SLOT(changed()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
boost::signals2::connection
|
||||||
|
FileMonitor2::connect(sig::slot_type const & slot)
|
||||||
|
{
|
||||||
|
return fileChanged_.connect(slot);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitor2::disconnect()
|
||||||
|
{
|
||||||
|
fileChanged_.disconnect_all_slots();
|
||||||
|
QObject::disconnect(this, SIGNAL(fileChanged()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitor2::changed()
|
||||||
|
{
|
||||||
|
// emit boost signal
|
||||||
|
fileChanged_();
|
||||||
|
Q_EMIT fileChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitorBlocker FileMonitor2::block(int delay)
|
||||||
|
{
|
||||||
|
FileMonitorBlocker blocker = blocker_.lock();
|
||||||
|
if (!blocker)
|
||||||
|
blocker_ = blocker = make_shared<FileMonitorBlockerGuard>(this);
|
||||||
|
blocker->setDelay(delay);
|
||||||
|
return blocker;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitorBlockerGuard::FileMonitorBlockerGuard(FileMonitor2 * parent)
|
||||||
|
: QObject(parent), parent_(parent), delay_(0)
|
||||||
|
{
|
||||||
|
QObject::disconnect(parent_->monitor_.get(), SIGNAL(fileChanged()),
|
||||||
|
parent_, SLOT(changed()));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void FileMonitorBlockerGuard::setDelay(int delay)
|
||||||
|
{
|
||||||
|
delay_ = max(delay_, delay);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileMonitorBlockerGuard::~FileMonitorBlockerGuard()
|
||||||
|
{
|
||||||
|
// closures can only copy local copies
|
||||||
|
FileMonitor2 * parent = parent_;
|
||||||
|
// parent is also our QObject::parent() so we are deleted before parent.
|
||||||
|
// Even if delay_ is 0, the QTimer is necessary. Indeed, the notifications
|
||||||
|
// from QFileSystemWatcher that we meant to ignore are not going to be
|
||||||
|
// treated immediately, so we must yield to give us the opportunity to
|
||||||
|
// ignore them.
|
||||||
|
QTimer::singleShot(delay_, parent, [parent]() {
|
||||||
|
parent->connectToFileMonitorGuard();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace support
|
||||||
|
} // namespace lyx
|
||||||
|
|
||||||
|
#include "moc_FileMonitor2.cpp"
|
192
src/support/FileMonitor2.h
Normal file
192
src/support/FileMonitor2.h
Normal file
@ -0,0 +1,192 @@
|
|||||||
|
// -*- C++ -*-
|
||||||
|
/**
|
||||||
|
* \file FileMonitor2.h
|
||||||
|
* This file is part of LyX, the document processor.
|
||||||
|
* Licence details can be found in the file COPYING.
|
||||||
|
*
|
||||||
|
* \author Guillaume Munch
|
||||||
|
*
|
||||||
|
* Full author contact details are available in file CREDITS.
|
||||||
|
*
|
||||||
|
* FileMonitor monitors a file and informs a listener when that file has
|
||||||
|
* changed.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef FILEMONITOR2_H
|
||||||
|
#define FILEMONITOR2_H
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include <QFileSystemWatcher>
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
#include <boost/signals2.hpp>
|
||||||
|
|
||||||
|
|
||||||
|
namespace lyx {
|
||||||
|
namespace support {
|
||||||
|
|
||||||
|
class FileName;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// FileMonitor2, a file monitor based on QFileSystemWatcher
|
||||||
|
///
|
||||||
|
|
||||||
|
class FileMonitor2;
|
||||||
|
class FileMonitorGuard;
|
||||||
|
using FileMonitorPtr = std::unique_ptr<FileMonitor2>;
|
||||||
|
|
||||||
|
///
|
||||||
|
/// Watch a file:
|
||||||
|
/// FileMonitorPtr monitor = FileSystemWatcher::monitor(file_with_path);
|
||||||
|
/// monitor.connect(...); //(using boost::signals2), or:
|
||||||
|
/// connect(monitor, SIGNAL(fileChanged()),...); // (using Qt)
|
||||||
|
///
|
||||||
|
/// Remember that a unique_ptr is automatically deleted at the end of a scope if
|
||||||
|
/// it has not been moved, or when assigned. When that happens, the signal
|
||||||
|
/// object is deleted and therefore all the connections are closed. The file
|
||||||
|
/// ceases being tracked when all the monitors for a file have been deleted.
|
||||||
|
///
|
||||||
|
/// Stop watching:
|
||||||
|
/// * as determined statically by the scope, or
|
||||||
|
/// * dynamically, using:
|
||||||
|
/// monitor = nullptr;
|
||||||
|
///
|
||||||
|
/// Watch a different file:
|
||||||
|
/// monitor = FileSystemWatcher::monitor(file_with_path2);
|
||||||
|
/// monitor.connect(...);
|
||||||
|
/// (stops watching the first)
|
||||||
|
///
|
||||||
|
/// Block notifications for the duration of a scope:
|
||||||
|
/// {
|
||||||
|
/// FileMonitorBlocker block = monitor.block();
|
||||||
|
/// ...
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// Reset connections:
|
||||||
|
/// monitor.disconnect();
|
||||||
|
/// or the disconnect method of the connection object for the boost signal.
|
||||||
|
///
|
||||||
|
class FileSystemWatcher
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// as described above
|
||||||
|
static FileMonitorPtr monitor(FileName const & file_with_path);
|
||||||
|
// Output whether the paths tracked by qwatcher_ and the active
|
||||||
|
// FileMonitorGuards are in correspondence.
|
||||||
|
static void debug();
|
||||||
|
private:
|
||||||
|
FileSystemWatcher();
|
||||||
|
// A global instance is created automatically on first call to monitor
|
||||||
|
static FileSystemWatcher & instance();
|
||||||
|
// Caches the monitor guards but allow them to be destroyed
|
||||||
|
std::map<std::string, std::weak_ptr<FileMonitorGuard>> store_;
|
||||||
|
// This class is a wrapper for QFileSystemWatcher
|
||||||
|
std::unique_ptr<QFileSystemWatcher> const qwatcher_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Must be unique per path
|
||||||
|
// Ends the watch when deleted
|
||||||
|
class FileMonitorGuard : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
/// Start the watch
|
||||||
|
FileMonitorGuard(std::string const & filename,
|
||||||
|
QFileSystemWatcher * qwatcher);
|
||||||
|
/// End the watch
|
||||||
|
~FileMonitorGuard();
|
||||||
|
/// absolute path being tracked
|
||||||
|
std::string const & filename() { return filename_; }
|
||||||
|
/// Make sure it is being monitored, after e.g. a deletion. See
|
||||||
|
/// <https://bugreports.qt.io/browse/QTBUG-46483>. This is called
|
||||||
|
/// automatically.
|
||||||
|
/// \param new_file If true, emit fileChanged if the file exists and was
|
||||||
|
/// successfully added.
|
||||||
|
void refresh(bool new_file = false);
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
/// Connect to this to be notified when the file changes
|
||||||
|
void fileChanged() const;
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
/// Receive notifications from the QFileSystemWatcher
|
||||||
|
void notifyChange(QString const & path);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::string const filename_;
|
||||||
|
QFileSystemWatcher * qwatcher_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class FileMonitorBlockerGuard : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
FileMonitor2 * parent_;
|
||||||
|
int delay_;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FileMonitorBlockerGuard(FileMonitor2 * parent);
|
||||||
|
~FileMonitorBlockerGuard();
|
||||||
|
void setDelay(int delay);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
using FileMonitorBlocker = std::shared_ptr<FileMonitorBlockerGuard>;
|
||||||
|
|
||||||
|
|
||||||
|
/// Main class
|
||||||
|
class FileMonitor2 : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
friend class FileMonitorBlockerGuard;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FileMonitor2(std::shared_ptr<FileMonitorGuard> monitor);
|
||||||
|
|
||||||
|
using sig = boost::signals2::signal<void()>;
|
||||||
|
/// Connect and you'll be informed when the file has changed.
|
||||||
|
boost::signals2::connection connect(sig::slot_type const &);
|
||||||
|
/// disconnect all slots connected to the boost signal fileChanged_ or to
|
||||||
|
/// the qt signal fileChanged()
|
||||||
|
void disconnect();
|
||||||
|
/// absolute path being tracked
|
||||||
|
std::string const & filename() { return monitor_->filename(); }
|
||||||
|
/// Creates a guard that blocks notifications. Copyable. Notifications from
|
||||||
|
/// this monitor are blocked as long as there are copies around.
|
||||||
|
/// \param delay is the amount waited in ms after expiration of the guard
|
||||||
|
/// before reconnecting. This delay thing is to deal with asynchronous
|
||||||
|
/// notifications in a not so elegant fashion. But it can also be used to
|
||||||
|
/// slow down incoming events.
|
||||||
|
FileMonitorBlocker block(int delay = 0);
|
||||||
|
/// Make sure the good file is being monitored, after e.g. a move or a
|
||||||
|
/// deletion. See <https://bugreports.qt.io/browse/QTBUG-46483>. This is
|
||||||
|
/// called automatically.
|
||||||
|
void refresh() { return monitor_->refresh(); }
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
/// Connect to this to be notified when the file changes
|
||||||
|
void fileChanged() const;
|
||||||
|
|
||||||
|
private Q_SLOTS:
|
||||||
|
/// Receive notifications from the FileMonitorGuard
|
||||||
|
void changed();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void connectToFileMonitorGuard();
|
||||||
|
// boost signal
|
||||||
|
sig fileChanged_;
|
||||||
|
// the unique watch for our file
|
||||||
|
std::shared_ptr<FileMonitorGuard> const monitor_;
|
||||||
|
//
|
||||||
|
std::weak_ptr<FileMonitorBlockerGuard> blocker_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace support
|
||||||
|
} // namespace lyx
|
||||||
|
|
||||||
|
#endif // FILEMONITOR2_H
|
@ -11,6 +11,7 @@ noinst_LIBRARIES = liblyxsupport.a
|
|||||||
|
|
||||||
MOCHEADER = \
|
MOCHEADER = \
|
||||||
ConsoleApplicationPrivate.h \
|
ConsoleApplicationPrivate.h \
|
||||||
|
FileMonitor2.h \
|
||||||
SystemcallPrivate.h
|
SystemcallPrivate.h
|
||||||
|
|
||||||
MOCEDFILES = $(MOCHEADER:%.h=moc_%.cpp)
|
MOCEDFILES = $(MOCHEADER:%.h=moc_%.cpp)
|
||||||
@ -33,6 +34,8 @@ AM_CPPFLAGS += -I$(srcdir)/.. \
|
|||||||
liblyxsupport_a_SOURCES = \
|
liblyxsupport_a_SOURCES = \
|
||||||
FileMonitor.h \
|
FileMonitor.h \
|
||||||
FileMonitor.cpp \
|
FileMonitor.cpp \
|
||||||
|
FileMonitor2.h \
|
||||||
|
FileMonitor2.cpp \
|
||||||
RandomAccessList.h \
|
RandomAccessList.h \
|
||||||
bind.h \
|
bind.h \
|
||||||
Cache.h \
|
Cache.h \
|
||||||
|
Loading…
x
Reference in New Issue
Block a user