kdelibs/kdecore/tests/kdirwatch_unittest.cpp

681 lines
22 KiB
C++
Raw Normal View History

2014-11-13 01:04:59 +02:00
/* This file is part of the KDE libraries
Copyright (c) 2009 David Faure <faure@kde.org>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include <kstandarddirs.h>
#include <kconfiggroup.h>
#include <QDir>
#include <kdebug.h>
#include <ktempdir.h>
#include <qtest_kde.h>
#include <qtestevent.h>
#include <kdirwatch.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <time.h>
#include <kde_file.h>
// Debugging notes: to see which inotify signals are emitted, either set s_verboseDebug=true
// at the top of kdirwatch.cpp, or use the command-line tool "inotifywait -m /path"
// Note that kdirlistertest and kdirmodeltest also exercise KDirWatch quite a lot.
static const char* methodToString(KDirWatch::Method method)
{
switch (method) {
case KDirWatch::FAM:
return "Fam";
case KDirWatch::QFSWatch:
return "QFSWatch";
default:
return "ERROR!";
}
}
class KDirWatch_UnitTest : public QObject
{
Q_OBJECT
public:
KDirWatch_UnitTest()
{
m_path = m_tempDir.name();
Q_ASSERT(m_path.endsWith('/'));
KConfigGroup config(KGlobal::config(), "DirWatch");
const QByteArray testMethod = qgetenv("KDIRWATCHTEST_METHOD");
if (!testMethod.isEmpty()) {
config.writeEntry("PreferredMethod", testMethod);
} else {
config.deleteEntry("PreferredMethod");
}
// Speed up the test by making the kdirwatch timer (to compress changes) faster
config.writeEntry("PollInterval", 50);
KDirWatch foo;
m_slow = (foo.internalMethod() == KDirWatch::FAM);
2014-11-13 01:04:59 +02:00
kDebug() << "Using method" << methodToString(foo.internalMethod());
}
private Q_SLOTS: // test methods
void initTestCase() {
// By creating the files upfront, we save waiting a full second for an mtime change
createFile(m_path + "ExistingFile");
createFile(m_path + "TestFile");
createFile(m_path + "nested_0");
createFile(m_path + "nested_1");
}
void touchOneFile();
void touch1000Files();
void watchAndModifyOneFile();
void removeAndReAdd();
void watchNonExistent();
void watchNonExistentWithSingleton();
void testDelete();
void testDeleteAndRecreateFile();
void testDeleteAndRecreateDir();
void testMoveTo();
void nestedEventLoop();
void testHardlinkChange();
void stopAndRestart();
protected Q_SLOTS: // internal slots
void nestedEventLoopSlot();
private:
void waitUntilMTimeChange(const QString& path);
void waitUntilNewSecond();
void waitUntilAfter(time_t ctime);
QList<QVariantList> waitForDirtySignal(KDirWatch& watch, int expected);
QList<QVariantList> waitForCreatedSignal(KDirWatch& watch, int expected);
QList<QVariantList> waitForDeletedSignal(KDirWatch& watch, int expected);
bool waitForOneSignal(KDirWatch& watch, const char* sig, const QString& path);
void createFile(const QString& path);
QString createFile(int num);
void removeFile(int num);
void appendToFile(const QString& path);
void appendToFile(int num);
KTempDir m_tempDir;
QString m_path;
bool m_slow;
};
QTEST_KDEMAIN_CORE(KDirWatch_UnitTest)
// Just to make the inotify packets bigger
static const char s_filePrefix[] = "This_is_a_test_file_";
static const int s_maxTries = 50;
// helper method: create a file
void KDirWatch_UnitTest::createFile(const QString& path)
{
QFile file(path);
bool ok = file.open(QIODevice::WriteOnly);
// prevent a compiler warning, marvelous isn't it?
if (!ok) {
Q_ASSERT(ok);
}
2014-11-13 01:04:59 +02:00
file.write(QByteArray("foo"));
file.close();
//kDebug() << path;
}
// helper method: create a file (identified by number)
QString KDirWatch_UnitTest::createFile(int num)
{
const QString fileName = s_filePrefix + QString::number(num);
createFile(m_path + fileName);
return m_path + fileName;
}
// helper method: delete a file (identified by number)
void KDirWatch_UnitTest::removeFile(int num)
{
const QString fileName = s_filePrefix + QString::number(num);
QFile::remove(m_path + fileName);
}
static QByteArray printableTime(time_t mtime)
{
struct tm* tmp = localtime(&mtime);
char outstr[200];
if (strftime(outstr, sizeof(outstr), "%T", tmp) == 0)
return "ERROR calling strftime!";
return outstr;
}
void KDirWatch_UnitTest::waitUntilMTimeChange(const QString& path)
{
// Wait until the current second is more than the file's mtime
// otherwise this change will go unnoticed
KDE_struct_stat stat_buf;
QCOMPARE(KDE::stat(path, &stat_buf), 0);
const time_t ctime = qMax(stat_buf.st_ctime, stat_buf.st_mtime);
waitUntilAfter(ctime);
}
void KDirWatch_UnitTest::waitUntilNewSecond()
{
struct timeval now_tv;
gettimeofday(&now_tv, NULL);
waitUntilAfter(now_tv.tv_sec);
}
void KDirWatch_UnitTest::waitUntilAfter(time_t ctime)
{
int totalWait = 0;
struct timeval now_tv;
Q_FOREVER {
gettimeofday(&now_tv, NULL);
// The mtime only has a granularity of a second, that's the whole issue;
// We can't just QTest::qWait(now_tv.tv_sec - stat_buf.st_ctime), that would
// be a full second every time.
if (now_tv.tv_sec == ctime) {
totalWait += 50;
QTest::qWait(50);
} else {
QVERIFY(now_tv.tv_sec > ctime); // can't go back in time ;)
QTest::qWait(50); // be safe
break;
}
}
//if (totalWait > 0)
kDebug() << "Waited" << totalWait << "ms so that now" << printableTime(now_tv.tv_sec) << "is >" << printableTime(ctime);
}
// helper method: modifies a file
void KDirWatch_UnitTest::appendToFile(const QString& path)
{
QVERIFY(QFile::exists(path));
waitUntilMTimeChange(path);
//const QString directory = QDir::cleanPath(path+"/..");
//waitUntilMTimeChange(directory);
QFile file(path);
QVERIFY(file.open(QIODevice::Append | QIODevice::WriteOnly));
file.write(QByteArray("foobar"));
file.close();
if (0) {
KDE_struct_stat stat_buf;
QCOMPARE(KDE::stat(path, &stat_buf), 0);
kDebug() << "After append: file ctime=" << printableTime(stat_buf.st_ctime);
QCOMPARE(KDE::stat(path, &stat_buf), 0);
kDebug() << "After append: directory mtime=" << printableTime(stat_buf.st_ctime);
}
}
// helper method: modifies a file (identified by number)
void KDirWatch_UnitTest::appendToFile(int num)
{
const QString fileName = s_filePrefix + QString::number(num);
appendToFile(m_path + fileName);
}
static QString removeTrailingSlash(const QString& path)
{
if (path.endsWith('/')) {
return path.left(path.length()-1);
} else {
return path;
}
}
// helper method
QList<QVariantList> KDirWatch_UnitTest::waitForDirtySignal(KDirWatch& watch, int expected)
{
QSignalSpy spyDirty(&watch, SIGNAL(dirty(QString)));
int numTries = 0;
// Give time for KDirWatch to notify us
while (spyDirty.count() < expected) {
if (++numTries > s_maxTries) {
kWarning() << "Timeout waiting for KDirWatch. Got" << spyDirty.count() << "dirty() signals, expected" << expected;
return spyDirty;
}
QTest::qWait(50);
}
return spyDirty;
}
bool KDirWatch_UnitTest::waitForOneSignal(KDirWatch& watch, const char* sig, const QString& path)
{
const QString expectedPath = removeTrailingSlash(path);
while (true) {
QSignalSpy spyDirty(&watch, sig);
int numTries = 0;
// Give time for KDirWatch to notify us
while (spyDirty.isEmpty()) {
if (++numTries > s_maxTries) {
kWarning() << "Timeout waiting for KDirWatch signal" << QByteArray(sig).mid(1) << "(" << path << ")";
return false;
}
QTest::qWait(50);
}
const QString got = spyDirty[0][0].toString();
if (got == expectedPath)
return true;
if (got.startsWith(expectedPath + '/')) {
kDebug() << "Ignoring notification of" << sig << '(' << got << ')';
2014-11-13 01:04:59 +02:00
continue;
}
kWarning() << "Expected" << sig << '(' << removeTrailingSlash(path) << ')' << "but got" << sig << '(' << got << ')';
return false;
}
}
QList<QVariantList> KDirWatch_UnitTest::waitForCreatedSignal(KDirWatch& watch, int expected)
{
QSignalSpy spyCreated(&watch, SIGNAL(created(QString)));
int numTries = 0;
// Give time for KDirWatch to notify us
while (spyCreated.count() < expected) {
if (++numTries > s_maxTries) {
kWarning() << "Timeout waiting for KDirWatch. Got" << spyCreated.count() << "created() signals, expected" << expected;
return spyCreated;
}
QTest::qWait(50);
}
return spyCreated;
}
QList<QVariantList> KDirWatch_UnitTest::waitForDeletedSignal(KDirWatch& watch, int expected)
{
QSignalSpy spyDeleted(&watch, SIGNAL(created(QString)));
int numTries = 0;
// Give time for KDirWatch to notify us
while (spyDeleted.count() < expected) {
if (++numTries > s_maxTries) {
kWarning() << "Timeout waiting for KDirWatch. Got" << spyDeleted.count() << "deleted() signals, expected" << expected;
return spyDeleted;
}
QTest::qWait(50);
}
return spyDeleted;
}
void KDirWatch_UnitTest::touchOneFile() // watch a dir, create a file in it
{
KDirWatch watch;
watch.addDir(m_path);
watch.startScan();
waitUntilMTimeChange(m_path);
// dirty(the directory) should be emitted.
const QString file0 = createFile(0);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
removeFile(0);
}
void KDirWatch_UnitTest::touch1000Files()
{
KDirWatch watch;
watch.addDir(m_path);
watch.startScan();
waitUntilMTimeChange(m_path);
const int fileCount = 100;
for (int i = 0; i < fileCount; ++i) {
createFile(i);
}
QList<QVariantList> spy = waitForDirtySignal(watch, fileCount);
// More stupid backends just see one mtime change on the directory
QVERIFY(spy.count() >= 1);
2014-11-13 01:04:59 +02:00
for (int i = 0; i < fileCount; ++i) {
removeFile(i);
}
}
void KDirWatch_UnitTest::watchAndModifyOneFile() // watch a specific file, and modify it
{
KDirWatch watch;
const QString existingFile = m_path + "ExistingFile";
watch.addFile(existingFile);
watch.startScan();
if (m_slow)
waitUntilNewSecond();
appendToFile(existingFile);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), existingFile));
}
void KDirWatch_UnitTest::removeAndReAdd()
{
KDirWatch watch;
watch.addDir(m_path);
watch.startScan();
if (m_slow)
waitUntilNewSecond(); // necessary for FAM
const QString file0 = createFile(0);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
// Just like KDirLister does: remove the watch, then re-add it.
watch.removeDir(m_path);
watch.addDir(m_path);
waitUntilMTimeChange(m_path); // necessary for FAM and QFSWatcher
2014-11-13 01:04:59 +02:00
const QString file1 = createFile(1);
//kDebug() << "created" << file1;
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
}
void KDirWatch_UnitTest::watchNonExistent()
{
KDirWatch watch;
// Watch "subdir", that doesn't exist yet
const QString subdir = m_path + "subdir";
QVERIFY(!QFile::exists(subdir));
watch.addDir(subdir);
watch.startScan();
if (m_slow)
waitUntilNewSecond();
// Now create it, KDirWatch should emit created()
kDebug() << "Creating" << subdir;
QDir().mkdir(subdir);
QVERIFY(waitForOneSignal(watch, SIGNAL(created(QString)), subdir));
KDirWatch::statistics();
// Play with addDir/removeDir, just for fun
watch.addDir(subdir);
watch.removeDir(subdir);
watch.addDir(subdir);
// Now watch files that don't exist yet
const QString file = subdir + "/0";
watch.addFile(file); // doesn't exist yet
const QString file1 = subdir + "/1";
watch.addFile(file1); // doesn't exist yet
watch.removeFile(file1); // forget it again
KDirWatch::statistics();
QVERIFY(!QFile::exists(file));
// Now create it, KDirWatch should emit created
kDebug() << "Creating" << file;
createFile(file);
QVERIFY(waitForOneSignal(watch, SIGNAL(created(QString)), file));
appendToFile(file);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), file));
// Create the file after all; we're not watching for it, but the dir will emit dirty
createFile(file1);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), subdir));
}
void KDirWatch_UnitTest::watchNonExistentWithSingleton()
{
const QString file = "/root/.ssh/authorized_keys";
KDirWatch::self()->addFile(file);
// When running this test in KDIRWATCHTEST_METHOD=QFSWatch, or when FAM is not available
// and we fallback to qfswatch, we end up creating the fsWatch in the kdirwatch singleton.
// Bug 261541 discovered that Qt hanged when deleting fsWatch once QCoreApp was gone,
// this is what this test is about.
2014-11-13 01:04:59 +02:00
}
void KDirWatch_UnitTest::testDelete()
{
const QString file1 = m_path + "del";
if (!QFile::exists(file1))
createFile(file1);
waitUntilMTimeChange(file1);
// Watch the file, then delete it, KDirWatch will emit deleted (and possibly dirty for the dir, if mtime changed)
KDirWatch watch;
watch.addFile(file1);
KDirWatch::statistics();
QSignalSpy spyDirty(&watch, SIGNAL(dirty(QString)));
QFile::remove(file1);
QVERIFY(waitForOneSignal(watch, SIGNAL(deleted(QString)), file1));
QTest::qWait(40); // just in case delayed processing would emit it
QCOMPARE(spyDirty.count(), 0);
}
void KDirWatch_UnitTest::testDeleteAndRecreateFile() // Useful for /etc/localtime for instance
{
const QString subdir = m_path + "subdir";
QDir().mkdir(subdir);
const QString file1 = subdir + "/1";
if (!QFile::exists(file1))
createFile(file1);
waitUntilMTimeChange(file1);
// Watch the file, then delete it, KDirWatch will emit deleted (and possibly dirty for the dir, if mtime changed)
KDirWatch watch;
watch.addFile(file1);
//KDE_struct_stat stat_buf;
//QCOMPARE(KDE::stat(QFile::encodeName(file1), &stat_buf), 0);
//kDebug() << "initial inode" << stat_buf.st_ino;
QFile::remove(file1);
// And recreate immediately, to try and fool KDirWatch with unchanged ctime/mtime ;)
// (This emulates the /etc/localtime case)
createFile(file1);
// gamin does not signal the change in this case; probably because it uses polling internally...
if (watch.internalMethod() == KDirWatch::FAM) {
QSKIP("Deleting and recreating a file is not detected by FAM (at least with gamin)", SkipAll);
2014-11-13 01:04:59 +02:00
}
//QCOMPARE(KDE::stat(QFile::encodeName(file1), &stat_buf), 0);
//kDebug() << "new inode" << stat_buf.st_ino; // same!
// NOTE: QFSWatch does not emit these
// QVERIFY(waitForOneSignal(watch, SIGNAL(deleted(QString)), file1));
// QVERIFY(waitForOneSignal(watch, SIGNAL(created(QString)), file1));
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), file1));
2014-11-13 01:04:59 +02:00
// QFileSystemWatcher, as documented, stops watching when the file is deleted
// so the appendToFile below will fail. Or further changes to /etc/localtime...
if (watch.internalMethod() == KDirWatch::QFSWatch) {
QSKIP("Limitation of QFSWatcher: it stops watching when deleting+recreating the file", SkipAll);
}
waitUntilMTimeChange(file1);
appendToFile(file1);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), file1));
}
void KDirWatch_UnitTest::testDeleteAndRecreateDir()
{
// Like KDirModelTest::testOverwriteFileWithDir does at the end.
// The linux-2.6.31 bug made kdirwatch emit deletion signals about the -new- dir!
KTempDir* tempDir1 = new KTempDir(KStandardDirs::locateLocal("tmp", "olddir-"));
KDirWatch watch;
const QString path1 = tempDir1->name();
watch.addDir(path1);
delete tempDir1;
KTempDir* tempDir2 = new KTempDir(KStandardDirs::locateLocal("tmp", "newdir-"));
const QString path2 = tempDir2->name();
watch.addDir(path2);
QVERIFY(waitForOneSignal(watch, SIGNAL(deleted(QString)), path1));
delete tempDir2;
}
void KDirWatch_UnitTest::testMoveTo()
{
// This reproduces the famous digikam crash, #222974
// A watched file was being rewritten (overwritten by ksavefile),
// which gives notifications "moved_to" followed by "delete_self"
2014-11-13 01:04:59 +02:00
//
// What happened then was that the delayed slotRescan
// would adjust things, making it status==Normal but the entry was
// listed as a "non-existent sub-entry" for the parent directory.
// That's inconsistent, and after removeFile() a dangling sub-entry would be left.
// Initial data: creating file subdir/1
const QString file1 = m_path + "moveTo";
createFile(file1);
KDirWatch watch;
watch.addDir(m_path);
watch.addFile(file1);
watch.startScan();
waitUntilMTimeChange(m_path);
2014-11-13 01:04:59 +02:00
// Atomic rename of "temp" to "file1", much like KAutoSave would do when saving file1 again
const QString filetemp = m_path + "temp";
createFile(filetemp);
QVERIFY(KDE::rename(filetemp, file1) == 0); // overwrite file1 with the tempfile
kDebug() << "Overwrite file1 with tempfile";
QSignalSpy spyCreated(&watch, SIGNAL(created(QString)));
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
// make sure we're still watching it
appendToFile(file1);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), file1));
//kDebug() << "after created";
//KDirWatch::statistics();
watch.removeFile(file1); // now we remove it
//kDebug() << "after removeFile";
//KDirWatch::statistics();
// Just touch another file to trigger a findSubEntry - this where the crash happened
waitUntilMTimeChange(m_path);
createFile(filetemp);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
}
void KDirWatch_UnitTest::nestedEventLoop() // #220153: watch two files, and modify 2nd while in slot for 1st
{
KDirWatch watch;
const QString file0 = m_path + "nested_0";
watch.addFile(file0);
const QString file1 = m_path + "nested_1";
watch.addFile(file1);
watch.startScan();
if (m_slow)
waitUntilNewSecond();
appendToFile(file0);
// use own spy, to connect it before nestedEventLoopSlot, otherwise it reverses order
QSignalSpy spyDirty(&watch, SIGNAL(dirty(QString)));
connect(&watch, SIGNAL(dirty(QString)), this, SLOT(nestedEventLoopSlot()));
waitForDirtySignal(watch, 1);
QVERIFY(spyDirty.count() >= 2);
QCOMPARE(spyDirty[0][0].toString(), file0);
QCOMPARE(spyDirty[spyDirty.count()-1][0].toString(), file1);
}
void KDirWatch_UnitTest::nestedEventLoopSlot()
{
const KDirWatch* const_watch = qobject_cast<const KDirWatch *>(sender());
KDirWatch* watch = const_cast<KDirWatch *>(const_watch);
// let's not come in this slot again
disconnect(watch, SIGNAL(dirty(QString)), this, SLOT(nestedEventLoopSlot()));
const QString file1 = m_path + "nested_1";
appendToFile(file1);
//kDebug() << "now waiting for signal";
// The nested event processing here was from a messagebox in #220153
QList<QVariantList> spy = waitForDirtySignal(*watch, 1);
QVERIFY(spy.count() >= 1);
QCOMPARE(spy[spy.count()-1][0].toString(), file1);
//kDebug() << "done";
// Now the user pressed reload...
const QString file0 = m_path + "nested_0";
watch->removeFile(file0);
watch->addFile(file0);
}
void KDirWatch_UnitTest::testHardlinkChange()
{
#ifdef Q_OS_UNIX
// The unittest for the "detecting hardlink change to /etc/localtime" problem
// described on kde-core-devel (2009-07-03).
// It shows that watching a specific file doesn't inform us that the file is
// being recreated. Better watch the directory, for that.
const QString existingFile = m_path + "ExistingFile";
KDirWatch watch;
watch.addFile(existingFile);
watch.startScan();
//waitUntilMTimeChange(existingFile);
//waitUntilMTimeChange(m_path);
QFile::remove(existingFile);
const QString testFile = m_path + "TestFile";
::link(QFile::encodeName(testFile), QFile::encodeName(existingFile)); // make ExistingFile "point" to TestFile
QVERIFY(QFile::exists(existingFile));
//QVERIFY(waitForOneSignal(watch, SIGNAL(deleted(QString)), existingFile));
if (watch.internalMethod() == KDirWatch::FAM)
2014-11-13 01:04:59 +02:00
QVERIFY(waitForOneSignal(watch, SIGNAL(created(QString)), existingFile));
else
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), existingFile));
//KDirWatch::statistics();
appendToFile(existingFile);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), existingFile));
#else
QSKIP("Unix-specific", SkipAll);
#endif
}
void KDirWatch_UnitTest::stopAndRestart()
{
KDirWatch watch;
watch.addDir(m_path);
watch.startScan();
waitUntilMTimeChange(m_path);
watch.stopDirScan(m_path);
const QString file0 = createFile(0);
QSignalSpy spyDirty(&watch, SIGNAL(dirty(QString)));
QTest::qWait(200);
QCOMPARE(spyDirty.count(), 0);// suspended -> no signal
watch.restartDirScan(m_path);
const QString file1 = createFile(1);
QVERIFY(waitForOneSignal(watch, SIGNAL(dirty(QString)), m_path));
removeFile(0);
removeFile(1);
}
#include "kdirwatch_unittest.moc"