File indexing completed on 2024-09-15 11:59:35
0001 /* 0002 This file is part of the KDE project 0003 SPDX-FileCopyrightText: 2007 David Faure <faure@kde.org> 0004 0005 SPDX-License-Identifier: LGPL-2.0-or-later 0006 */ 0007 0008 #include "kdirlistertest.h" 0009 0010 #include "jobuidelegatefactory.h" 0011 #include "kiotesthelper.h" 0012 #include <kio/copyjob.h> 0013 #include <kio/deletejob.h> 0014 #include <kio/jobuidelegateextension.h> 0015 #include <kio/simplejob.h> 0016 #include <kprotocolinfo.h> 0017 0018 #include <KDirWatch> 0019 0020 #include <QDebug> 0021 #include <QTemporaryFile> 0022 #include <QTest> 0023 0024 QTEST_MAIN(KDirListerTest) 0025 0026 static constexpr bool s_workaroundBrokenInotify = false; 0027 0028 GlobalInits::GlobalInits() 0029 { 0030 // Must be done before the QSignalSpys connect 0031 qRegisterMetaType<KFileItem>(); 0032 qRegisterMetaType<KFileItemList>(); 0033 qRegisterMetaType<KIO::Job *>(); 0034 } 0035 0036 QString KDirListerTest::tempPath() const 0037 { 0038 return m_tempDir->path() + '/'; 0039 } 0040 0041 void KDirListerTest::initTestCase() 0042 { 0043 // To avoid failing on broken locally defined MIME types 0044 QStandardPaths::setTestModeEnabled(true); 0045 0046 m_tempDir.reset(new QTemporaryDir(homeTmpDir())); 0047 0048 // No message dialogs 0049 KIO::setDefaultJobUiDelegateExtension(nullptr); 0050 KIO::setDefaultJobUiDelegateFactoryV2(nullptr); 0051 0052 m_exitCount = 1; 0053 0054 s_referenceTimeStamp = QDateTime::currentDateTime().addSecs(-120); // 2 minutes ago 0055 0056 // Create test data: 0057 /* 0058 * PATH/toplevelfile_1 0059 * PATH/toplevelfile_2 0060 * PATH/toplevelfile_3 0061 * PATH/subdir 0062 * PATH/subdir/testfile 0063 * PATH/subdir/subsubdir 0064 * PATH/subdir/subsubdir/testfile 0065 */ 0066 createTestFile(tempPath() + "toplevelfile_1"); 0067 createTestFile(tempPath() + "toplevelfile_2"); 0068 createTestFile(tempPath() + "toplevelfile_3"); 0069 createTestDirectory(tempPath() + "subdir"); 0070 createTestDirectory(tempPath() + "subdir/subsubdir"); 0071 0072 qRegisterMetaType<QList<QPair<KFileItem, KFileItem>>>(); 0073 } 0074 0075 void KDirListerTest::cleanup() 0076 { 0077 m_dirLister.clearSpies(); 0078 disconnect(&m_dirLister, nullptr, this, nullptr); 0079 } 0080 0081 void KDirListerTest::testInvalidUrl() 0082 { 0083 m_dirLister.openUrl(QUrl(":/"), KDirLister::NoFlags); 0084 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0085 QVERIFY(m_dirLister.spyJobError.wait()); 0086 QCOMPARE(m_dirLister.spyJobError.at(0).at(0).value<KIO::Job *>()->error(), KIO::ERR_MALFORMED_URL); 0087 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0088 QVERIFY(m_dirLister.isFinished()); 0089 } 0090 0091 void KDirListerTest::testNonListableUrl() 0092 { 0093 m_dirLister.openUrl(QUrl("data:foo"), KDirLister::NoFlags); 0094 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0095 QVERIFY(m_dirLister.spyJobError.wait()); 0096 QCOMPARE(m_dirLister.spyJobError.at(0).at(0).value<KIO::Job *>()->error(), KIO::ERR_UNSUPPORTED_ACTION); 0097 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0098 QVERIFY(m_dirLister.isFinished()); 0099 } 0100 0101 void KDirListerTest::testOpenUrl() 0102 { 0103 m_items.clear(); 0104 const QString path = tempPath(); 0105 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0106 // The call to openUrl itself, emits started 0107 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0108 0109 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0110 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0111 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 0112 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0113 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0114 QCOMPARE(m_dirLister.spyClear.count(), 1); 0115 0116 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0117 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0118 #endif 0119 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0120 0121 QCOMPARE(m_dirLister.spyRedirection.count(), 0); 0122 QCOMPARE(m_items.count(), 0); 0123 QVERIFY(!m_dirLister.isFinished()); 0124 0125 // then wait for completed 0126 qDebug("waiting for completed"); 0127 QTRY_COMPARE(m_dirLister.spyStarted.count(), 1); 0128 QTRY_COMPARE(m_dirLister.spyCompleted.count(), 1); 0129 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0130 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0131 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0132 QCOMPARE(m_dirLister.spyClear.count(), 1); 0133 0134 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0135 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0136 #endif 0137 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0138 0139 QCOMPARE(m_dirLister.spyRedirection.count(), 0); 0140 // qDebug() << m_items; 0141 // qDebug() << "In dir" << QDir(path).entryList( QDir::AllEntries | QDir::NoDotAndDotDot); 0142 QCOMPARE(m_items.count(), fileCount()); 0143 QVERIFY(m_dirLister.isFinished()); 0144 disconnect(&m_dirLister, nullptr, this, nullptr); 0145 0146 const QString fileName = QStringLiteral("toplevelfile_3"); 0147 const QUrl itemUrl = QUrl::fromLocalFile(path + fileName); 0148 KFileItem byName = m_dirLister.findByName(fileName); 0149 QVERIFY(!byName.isNull()); 0150 QCOMPARE(byName.url().toString(), itemUrl.toString()); 0151 QCOMPARE(byName.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0152 KFileItem byUrl = m_dirLister.findByUrl(itemUrl); 0153 QVERIFY(!byUrl.isNull()); 0154 QCOMPARE(byUrl.url().toString(), itemUrl.toString()); 0155 QCOMPARE(byUrl.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0156 KFileItem itemForUrl = KDirLister::cachedItemForUrl(itemUrl); 0157 QVERIFY(!itemForUrl.isNull()); 0158 QCOMPARE(itemForUrl.url().toString(), itemUrl.toString()); 0159 QCOMPARE(itemForUrl.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0160 0161 KFileItem rootByUrl = m_dirLister.findByUrl(QUrl::fromLocalFile(path)); 0162 QVERIFY(!rootByUrl.isNull()); 0163 QCOMPARE(QString(rootByUrl.url().toLocalFile() + '/'), path); 0164 0165 m_dirLister.clearSpies(); // for the tests that call testOpenUrl for setup 0166 } 0167 0168 // This test assumes testOpenUrl was run before. So m_dirLister is holding the items already. 0169 void KDirListerTest::testOpenUrlFromCache() 0170 { 0171 // Do the same again, it should behave the same, even with the items in the cache 0172 testOpenUrl(); 0173 0174 // Get into the case where another dirlister is holding the items 0175 { 0176 m_items.clear(); 0177 const QString path = tempPath(); 0178 MyDirLister secondDirLister; 0179 connect(&secondDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0180 0181 secondDirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0182 QCOMPARE(secondDirLister.spyStarted.count(), 1); 0183 QCOMPARE(secondDirLister.spyCompleted.count(), 0); 0184 QCOMPARE(secondDirLister.spyCompletedQUrl.count(), 0); 0185 QCOMPARE(secondDirLister.spyCanceled.count(), 0); 0186 QCOMPARE(secondDirLister.spyCanceledQUrl.count(), 0); 0187 QCOMPARE(secondDirLister.spyClear.count(), 1); 0188 0189 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0190 QCOMPARE(secondDirLister.spyClearQUrl.count(), 0); 0191 #endif 0192 QCOMPARE(secondDirLister.spyClearDir.count(), 0); 0193 0194 QCOMPARE(m_items.count(), 0); 0195 QVERIFY(!secondDirLister.isFinished()); 0196 0197 // then wait for completed 0198 qDebug("waiting for completed"); 0199 QTRY_COMPARE(secondDirLister.spyStarted.count(), 1); 0200 QTRY_COMPARE(secondDirLister.spyCompleted.count(), 1); 0201 QCOMPARE(secondDirLister.spyCompletedQUrl.count(), 1); 0202 QCOMPARE(secondDirLister.spyCanceled.count(), 0); 0203 QCOMPARE(secondDirLister.spyCanceledQUrl.count(), 0); 0204 QCOMPARE(secondDirLister.spyClear.count(), 1); 0205 0206 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0207 QCOMPARE(secondDirLister.spyClearQUrl.count(), 0); 0208 #endif 0209 QCOMPARE(secondDirLister.spyClearDir.count(), 0); 0210 0211 QCOMPARE(m_items.count(), 4); 0212 QVERIFY(secondDirLister.isFinished()); 0213 } 0214 0215 disconnect(&m_dirLister, nullptr, this, nullptr); 0216 } 0217 0218 // This test assumes testOpenUrl was run before. So m_dirLister is holding the items already. 0219 // This test creates 1 file in the temporary directory 0220 void KDirListerTest::testNewItem() 0221 { 0222 QCOMPARE(m_items.count(), 4); 0223 const QString path = tempPath(); 0224 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0225 0226 qDebug() << "Creating a new file"; 0227 const QString fileName = QStringLiteral("toplevelfile_new"); 0228 0229 createSimpleFile(path + fileName); 0230 0231 QTRY_COMPARE(m_items.count(), 5); 0232 QCOMPARE(m_dirLister.spyStarted.count(), 1); // Updates call started 0233 QCOMPARE(m_dirLister.spyCompleted.count(), 1); // and completed 0234 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0235 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0236 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0237 QCOMPARE(m_dirLister.spyClear.count(), 0); 0238 0239 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0240 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0241 #endif 0242 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0243 0244 const QUrl itemUrl = QUrl::fromLocalFile(path + fileName); 0245 KFileItem itemForUrl = KDirLister::cachedItemForUrl(itemUrl); 0246 QVERIFY(!itemForUrl.isNull()); 0247 QCOMPARE(itemForUrl.url().toString(), itemUrl.toString()); 0248 QCOMPARE(itemForUrl.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0249 disconnect(&m_dirLister, nullptr, this, nullptr); 0250 } 0251 0252 // This test assumes testNewItem was run before. So m_dirLister is holding the items already. 0253 // This test creates 100 more files in the temporary directory in reverse order 0254 void KDirListerTest::testNewItems() 0255 { 0256 QCOMPARE(m_items.count(), 5); 0257 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0258 0259 const QString path = tempPath(); 0260 0261 qDebug() << "Creating 100 new files"; 0262 for (int i = 50; i > 0; i--) { 0263 createSimpleFile(path + QString("toplevelfile_new_%1").arg(i)); 0264 } 0265 QTest::qWait(1000); // Create them with 1s difference 0266 for (int i = 100; i > 50; i--) { 0267 createSimpleFile(path + QString("toplevelfile_new_%1").arg(i)); 0268 } 0269 0270 // choose one of the new created files 0271 const QString fileName = QStringLiteral("toplevelfile_new_50"); 0272 0273 QTRY_COMPARE(m_items.count(), 105); 0274 0275 QVERIFY(m_dirLister.spyStarted.count() > 0 && m_dirLister.spyStarted.count() < 3); // Updates call started, probably twice 0276 QVERIFY(m_dirLister.spyCompleted.count() > 0 && m_dirLister.spyCompleted.count() < 3); // and completed, probably twice 0277 QVERIFY(m_dirLister.spyCompletedQUrl.count() < 3); 0278 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0279 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0280 QCOMPARE(m_dirLister.spyClear.count(), 0); 0281 0282 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0283 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0284 #endif 0285 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0286 0287 const QUrl itemUrl = QUrl::fromLocalFile(path + fileName); 0288 KFileItem itemForUrl = KDirLister::cachedItemForUrl(itemUrl); 0289 QVERIFY(!itemForUrl.isNull()); 0290 QCOMPARE(itemForUrl.url().toString(), itemUrl.toString()); 0291 QCOMPARE(itemForUrl.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0292 } 0293 0294 void KDirListerTest::benchFindByUrl() 0295 { 0296 // The time used should be in the order of O(100*log2(100)) 0297 const QString path = tempPath(); 0298 QBENCHMARK { 0299 for (int i = 100; i > 0; i--) { 0300 KFileItem cachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(path + QString("toplevelfile_new_%1").arg(i))); 0301 QVERIFY(!cachedItem.isNull()); 0302 } 0303 } 0304 } 0305 0306 void KDirListerTest::testNewItemByCopy() 0307 { 0308 // This test creates a file using KIO::copyAs, like knewmenu.cpp does. 0309 // Useful for testing #192185, i.e. whether we catch the kdirwatch event and avoid 0310 // a KFileItem::refresh(). 0311 const int origItemCount = m_items.count(); 0312 const QString path = tempPath(); 0313 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0314 0315 QTest::qWait(1000); // We need a 1s timestamp difference on the dir, otherwise FAM won't notice anything. 0316 0317 const QString fileName = QStringLiteral("toplevelfile_copy"); 0318 const QUrl itemUrl = QUrl::fromLocalFile(path + fileName); 0319 KIO::CopyJob *job = KIO::copyAs(QUrl::fromLocalFile(path + "toplevelfile_3"), itemUrl, KIO::HideProgressInfo); 0320 job->exec(); 0321 0322 // Give time for KDirWatch/KDirNotify to notify us 0323 QTRY_COMPARE(m_items.count(), origItemCount + 1); 0324 0325 QCOMPARE(m_dirLister.spyStarted.count(), 1); // Updates call started 0326 QCOMPARE(m_dirLister.spyCompleted.count(), 1); // and completed 0327 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0328 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0329 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0330 QCOMPARE(m_dirLister.spyClear.count(), 0); 0331 0332 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0333 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0334 #endif 0335 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0336 0337 // Give some time to KDirWatch 0338 QTest::qWait(1000); 0339 0340 KFileItem itemForUrl = KDirLister::cachedItemForUrl(itemUrl); 0341 QVERIFY(!itemForUrl.isNull()); 0342 QCOMPARE(itemForUrl.url().toString(), itemUrl.toString()); 0343 QCOMPARE(itemForUrl.entry().stringValue(KIO::UDSEntry::UDS_NAME), fileName); 0344 } 0345 0346 void KDirListerTest::testNewItemByCopyInSubDir() // #440712 0347 { 0348 // Test that copying a file to a directory whose parent is listed 0349 // triggers refreshItems for the directory 0350 const int origItemCount = m_items.count(); 0351 const QString path = tempPath(); 0352 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0353 connect(&m_dirLister, &KDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems); 0354 QSignalSpy refreshItemSpy(this, &KDirListerTest::refreshItemsReceived); 0355 0356 const QUrl subDirUrl = QUrl::fromLocalFile(path + "subdir"); 0357 KFileItem itemForUrl = KDirLister::cachedItemForUrl(subDirUrl); 0358 auto origModificationDate = itemForUrl.entry().numberValue(KIO::UDSEntry::UDS_MODIFICATION_TIME); 0359 0360 const QString fileName = path + "subdir/toplevelfile_copy"; 0361 const QUrl itemUrl = QUrl::fromLocalFile(fileName); 0362 KIO::CopyJob *job = KIO::copyAs(QUrl::fromLocalFile(path + "toplevelfile_3"), itemUrl, KIO::HideProgressInfo); 0363 QVERIFY(job->exec()); 0364 0365 QTRY_COMPARE(m_items.count(), origItemCount); 0366 0367 // Give some time to KDirNotify 0368 QVERIFY(refreshItemSpy.wait(100)); 0369 0370 itemForUrl = KDirLister::cachedItemForUrl(subDirUrl); 0371 QVERIFY(itemForUrl.entry().numberValue(KIO::UDSEntry::UDS_MODIFICATION_TIME) > origModificationDate); 0372 0373 // clean leftover file 0374 QVERIFY(QFile(fileName).remove()); 0375 m_refreshedItems.clear(); 0376 } 0377 0378 void KDirListerTest::testNewItemsInSymlink() // #213799 0379 { 0380 const int origItemCount = m_items.count(); 0381 QCOMPARE(fileCount(), origItemCount); 0382 const QString path = tempPath(); 0383 QTemporaryFile tempFile; 0384 QVERIFY(tempFile.open()); 0385 const QString symPath = tempFile.fileName() + "_link"; 0386 tempFile.close(); 0387 const bool symlinkOk = KIOPrivate::createSymlink(path, symPath); 0388 if (!symlinkOk) { 0389 const QString error = 0390 QString::fromLatin1("Failed to create symlink '%1' pointing to '%2': %3").arg(symPath, path, QString::fromLocal8Bit(strerror(errno))); 0391 QVERIFY2(symlinkOk, qPrintable(error)); 0392 } 0393 MyDirLister dirLister2; 0394 m_items2.clear(); 0395 m_items.clear(); 0396 connect(&dirLister2, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems2); 0397 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0398 0399 // The initial listing 0400 m_dirLister.openUrl(QUrl::fromLocalFile(path)); 0401 dirLister2.openUrl(QUrl::fromLocalFile(symPath)); 0402 QTRY_COMPARE(m_items.count(), m_items2.count()); 0403 QTRY_COMPARE(m_items.count(), origItemCount); 0404 QTRY_VERIFY(dirLister2.isFinished()); 0405 0406 QTest::qWait(100); // We need a 1s timestamp difference on the dir, otherwise FAM won't notice anything. 0407 0408 qDebug() << "Creating new file"; 0409 const QString fileName = QStringLiteral("toplevelfile_newinlink"); 0410 createSimpleFile(path + fileName); 0411 0412 if (s_workaroundBrokenInotify) { 0413 org::kde::KDirNotify::emitFilesAdded(QUrl::fromLocalFile(path)); 0414 } 0415 0416 // Give time for KDirWatch to notify us 0417 QTRY_COMPARE(m_items2.count(), origItemCount + 1); 0418 QTRY_COMPARE(m_items.count(), origItemCount + 1); 0419 0420 // Now create an item using the symlink-path 0421 const QString fileName2 = QStringLiteral("toplevelfile_newinlink2"); 0422 { 0423 createSimpleFile(path + fileName2); 0424 0425 // Give time for KDirWatch to notify us 0426 QTRY_COMPARE(m_items2.count(), origItemCount + 2); 0427 QTRY_COMPARE(m_items.count(), origItemCount + 2); 0428 } 0429 QCOMPARE(fileCount(), m_items.count()); 0430 0431 // Test file deletion 0432 { 0433 qDebug() << "Deleting" << (path + fileName); 0434 QTest::qWait(100); // for timestamp difference 0435 QFile::remove(path + fileName); 0436 QTRY_COMPARE(dirLister2.spyItemsDeleted.count(), 1); 0437 QTRY_COMPARE(m_dirLister.spyItemsDeleted.count(), 1); 0438 const KFileItem item = dirLister2.spyItemsDeleted[0][0].value<KFileItemList>().at(0); 0439 QCOMPARE(item.url().toLocalFile(), QString(symPath + '/' + fileName)); 0440 } 0441 // Test file deletion in symlink dir #469254 0442 { 0443 qDebug() << "Deleting" << (symPath + "/" + fileName2); 0444 QTest::qWait(100); // for timestamp difference 0445 QFile::remove(symPath + "/" + fileName2); 0446 QTRY_COMPARE(dirLister2.spyItemsDeleted.count(), 2); 0447 QTRY_COMPARE(m_dirLister.spyItemsDeleted.count(), 2); 0448 const KFileItem item = dirLister2.spyItemsDeleted[1][0].value<KFileItemList>().at(0); 0449 QCOMPARE(item.url().toLocalFile(), QString(symPath + '/' + fileName2)); 0450 } 0451 0452 // TODO: test file update. 0453 disconnect(&m_dirLister, nullptr, this, nullptr); 0454 0455 QFile::remove(symPath); 0456 m_dirLister.openUrl(QUrl::fromLocalFile(tempPath())); 0457 } 0458 0459 // This test assumes testOpenUrl was run before. So m_dirLister is holding the items already. 0460 // Modifies one of the files to have html content 0461 void KDirListerTest::testRefreshItems() 0462 { 0463 m_refreshedItems.clear(); 0464 0465 const QString path = tempPath(); 0466 const QString fileName = path + "toplevelfile_1"; 0467 KFileItem cachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(fileName)); 0468 QVERIFY(!cachedItem.isNull()); 0469 QCOMPARE(cachedItem.mimetype(), QString("application/octet-stream")); 0470 0471 connect(&m_dirLister, &KCoreDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems); 0472 0473 QFile file(fileName); 0474 QVERIFY(file.open(QIODevice::Append)); 0475 file.write(QByteArray("<html>")); 0476 file.close(); 0477 QCOMPARE(QFileInfo(fileName).size(), 11LL /*Hello world*/ + 6 /*<html>*/); 0478 0479 QTRY_VERIFY(!m_refreshedItems.isEmpty()); 0480 0481 QCOMPARE(m_dirLister.spyStarted.count(), 0); // fast path: no directory listing needed 0482 QVERIFY(m_dirLister.spyCompleted.count() < 2); 0483 QVERIFY(m_dirLister.spyCompletedQUrl.count() < 2); 0484 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0485 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0486 QCOMPARE(m_dirLister.spyClear.count(), 0); 0487 0488 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0489 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0490 #endif 0491 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0492 0493 QCOMPARE(m_refreshedItems.count(), 1); 0494 QPair<KFileItem, KFileItem> entry = m_refreshedItems.first(); 0495 QCOMPARE(entry.first.url().toLocalFile(), fileName); 0496 QCOMPARE(entry.first.size(), KIO::filesize_t(11)); 0497 QCOMPARE(entry.first.mimetype(), QString("application/octet-stream")); 0498 QCOMPARE(entry.second.url().toLocalFile(), fileName); 0499 QCOMPARE(entry.second.size(), KIO::filesize_t(11 /*Hello world*/ + 6 /*<html>*/)); 0500 QCOMPARE(entry.second.mimetype(), QString("text/html")); 0501 0502 // Let's see what KDirLister has in cache now 0503 cachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(fileName)); 0504 QCOMPARE(cachedItem.size(), KIO::filesize_t(11 /*Hello world*/ + 6 /*<html>*/)); 0505 m_refreshedItems.clear(); 0506 } 0507 0508 // Refresh the root item, plus a hidden file, e.g. changing its icon. #190535 0509 void KDirListerTest::testRefreshRootItem() 0510 { 0511 // This test assumes testOpenUrl was run before. So m_dirLister is holding the items already. 0512 m_refreshedItems.clear(); 0513 m_refreshedItems2.clear(); 0514 0515 // The item will be the root item of dirLister2, but also a child item 0516 // of m_dirLister. 0517 // In #190535 it would show "." instead of the subdir name, after a refresh... 0518 const QString path = tempPath() + "subdir"; 0519 MyDirLister dirLister2; 0520 fillDirLister2(dirLister2, path); 0521 0522 // Change the subdir by creating a file in it 0523 waitUntilMTimeChange(path); 0524 const QString foobar = path + "/.foobar"; 0525 createSimpleFile(foobar); 0526 0527 connect(&m_dirLister, &KCoreDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems); 0528 0529 // Arguably, the mtime change of "subdir" should lead to a refreshItem of subdir in the root dir. 0530 // So the next line shouldn't be necessary, if KDirLister did this correctly. This isn't what this test is about though. 0531 org::kde::KDirNotify::emitFilesChanged(QList<QUrl>{QUrl::fromLocalFile(path)}); 0532 QTRY_VERIFY(!m_refreshedItems.isEmpty()); 0533 0534 QCOMPARE(m_dirLister.spyStarted.count(), 0); 0535 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0536 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 0537 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0538 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0539 QCOMPARE(m_dirLister.spyClear.count(), 0); 0540 0541 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0542 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0543 #endif 0544 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0545 0546 QCOMPARE(m_refreshedItems.count(), 1); 0547 QPair<KFileItem, KFileItem> entry = m_refreshedItems.first(); 0548 QCOMPARE(entry.first.url().toLocalFile(), path); 0549 QCOMPARE(entry.first.name(), QString("subdir")); 0550 QCOMPARE(entry.second.url().toLocalFile(), path); 0551 QCOMPARE(entry.second.name(), QString("subdir")); 0552 0553 QCOMPARE(m_refreshedItems2.count(), 1); 0554 entry = m_refreshedItems2.first(); 0555 QCOMPARE(entry.first.url().toLocalFile(), path); 0556 QCOMPARE(entry.second.url().toLocalFile(), path); 0557 // item name() doesn't matter here, it's the root item. 0558 0559 m_refreshedItems.clear(); 0560 m_refreshedItems2.clear(); 0561 0562 waitUntilMTimeChange(path); 0563 const QString directoryFile = path + "/.directory"; 0564 createSimpleFile(directoryFile); 0565 0566 org::kde::KDirNotify::emitFilesAdded(QUrl::fromLocalFile(path)); 0567 QTest::qWait(200); 0568 // The order of these two is not deterministic 0569 org::kde::KDirNotify::emitFilesChanged(QList<QUrl>{QUrl::fromLocalFile(directoryFile)}); 0570 org::kde::KDirNotify::emitFilesChanged(QList<QUrl>{QUrl::fromLocalFile(path)}); 0571 QTRY_VERIFY(!m_refreshedItems.isEmpty()); 0572 0573 QCOMPARE(m_refreshedItems.count(), 1); 0574 entry = m_refreshedItems.first(); 0575 QCOMPARE(entry.first.url().toLocalFile(), path); 0576 QCOMPARE(entry.second.url().toLocalFile(), path); 0577 0578 m_refreshedItems.clear(); 0579 m_refreshedItems2.clear(); 0580 0581 // Note: this test leaves the .directory file as a side effect. 0582 // Hidden though, shouldn't matter. 0583 } 0584 0585 void KDirListerTest::testDeleteItem() 0586 { 0587 testOpenUrl(); // ensure m_items is up-to-date 0588 0589 const int origItemCount = m_items.count(); 0590 QCOMPARE(fileCount(), origItemCount); 0591 const QString path = tempPath(); 0592 0593 // qDebug() << "Removing " << path+"toplevelfile_new"; 0594 QFile::remove(path + QString("toplevelfile_new")); 0595 // the remove() doesn't always trigger kdirwatch in stat mode, if this all happens in the same second 0596 KDirWatch::self()->setDirty(path); 0597 0598 // The signal should be emitted once with the deleted file 0599 QTRY_COMPARE(m_dirLister.spyItemsDeleted.count(), 1); 0600 0601 // OK now kdirlister told us the file was deleted, let's try a re-listing 0602 m_items.clear(); 0603 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0604 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0605 QVERIFY(!m_dirLister.isFinished()); 0606 0607 QTRY_COMPARE(m_items.count(), origItemCount - 1); 0608 QVERIFY(m_dirLister.isFinished()); 0609 0610 disconnect(&m_dirLister, nullptr, this, nullptr); 0611 QCOMPARE(fileCount(), m_items.count()); 0612 } 0613 0614 void KDirListerTest::testDeleteItems() 0615 { 0616 testOpenUrl(); // ensure m_items is up-to-date 0617 0618 const int origItemCount = m_items.count(); 0619 QCOMPARE(fileCount(), origItemCount); 0620 const QString path = tempPath(); 0621 0622 qDebug() << "Removing 100 files from " << path; 0623 for (int i = 0; i <= 100; ++i) { 0624 QFile::remove(path + QString("toplevelfile_new_%1").arg(i)); 0625 } 0626 // the remove() doesn't always trigger kdirwatch in stat mode, if this all happens in the same second 0627 KDirWatch::self()->setDirty(path); 0628 0629 // The signal could be emitted 1 time with all the deleted files or more times 0630 QTRY_VERIFY(m_dirLister.spyItemsDeleted.count() > 0); 0631 0632 // OK now kdirlister told us the file was deleted, let's try a re-listing 0633 m_items.clear(); 0634 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0635 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0636 QTRY_COMPARE(m_items.count(), origItemCount - 100); 0637 QVERIFY(m_dirLister.isFinished()); 0638 0639 disconnect(&m_dirLister, nullptr, this, nullptr); 0640 QCOMPARE(fileCount(), m_items.count()); 0641 } 0642 0643 void KDirListerTest::testRenameItem() 0644 { 0645 m_refreshedItems2.clear(); 0646 const QString dirPath = tempPath(); 0647 connect(&m_dirLister, &KCoreDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems2); 0648 const QString path = dirPath + "toplevelfile_2"; 0649 const QString newPath = dirPath + "toplevelfile_2.renamed.cpp"; 0650 0651 KIO::SimpleJob *job = KIO::rename(QUrl::fromLocalFile(path), QUrl::fromLocalFile(newPath), KIO::HideProgressInfo); 0652 QVERIFY(job->exec()); 0653 0654 QSignalSpy spyRefreshItems(&m_dirLister, &KCoreDirLister::refreshItems); 0655 QVERIFY(spyRefreshItems.wait(2000)); 0656 QTRY_COMPARE(m_refreshedItems2.count(), 1); 0657 QPair<KFileItem, KFileItem> entry = m_refreshedItems2.first(); 0658 QCOMPARE(entry.first.url().toLocalFile(), path); 0659 QCOMPARE(entry.first.mimetype(), QString("application/octet-stream")); 0660 QCOMPARE(entry.second.url().toLocalFile(), newPath); 0661 QCOMPARE(entry.second.mimetype(), QString("text/x-c++src")); 0662 disconnect(&m_dirLister, nullptr, this, nullptr); 0663 0664 // Let's see what KDirLister has in cache now 0665 KFileItem cachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(newPath)); 0666 QVERIFY(!cachedItem.isNull()); 0667 QCOMPARE(cachedItem.url().toLocalFile(), newPath); 0668 KFileItem oldCachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(path)); 0669 QVERIFY(oldCachedItem.isNull()); 0670 m_refreshedItems2.clear(); 0671 } 0672 0673 void KDirListerTest::testRenameAndOverwrite() // has to be run after testRenameItem 0674 { 0675 // Rename toplevelfile_2.renamed.html to toplevelfile_2, overwriting it. 0676 const QString dirPath = tempPath(); 0677 const QString path = dirPath + "toplevelfile_2"; 0678 createTestFile(path); 0679 0680 if (s_workaroundBrokenInotify) { 0681 org::kde::KDirNotify::emitFilesAdded(QUrl::fromLocalFile(dirPath)); 0682 } 0683 0684 KFileItem existingItem; 0685 while (existingItem.isNull()) { 0686 QTest::qWait(100); 0687 existingItem = m_dirLister.findByUrl(QUrl::fromLocalFile(path)); 0688 }; 0689 QCOMPARE(existingItem.url().toLocalFile(), path); 0690 0691 m_refreshedItems.clear(); 0692 connect(&m_dirLister, &KCoreDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems); 0693 const QString newPath = dirPath + "toplevelfile_2.renamed.cpp"; 0694 0695 KIO::SimpleJob *job = KIO::rename(QUrl::fromLocalFile(newPath), QUrl::fromLocalFile(path), KIO::Overwrite | KIO::HideProgressInfo); 0696 bool ok = job->exec(); 0697 QVERIFY(ok); 0698 0699 if (m_refreshedItems.isEmpty()) { 0700 QTRY_VERIFY(!m_refreshedItems.isEmpty()); // could come from KDirWatch or KDirNotify. 0701 } 0702 0703 // Check that itemsDeleted was emitted -- preferably BEFORE refreshItems, 0704 // but we can't easily check that with QSignalSpy... 0705 QCOMPARE(m_dirLister.spyItemsDeleted.count(), 1); 0706 0707 QCOMPARE(m_refreshedItems.count(), 1); 0708 QPair<KFileItem, KFileItem> entry = m_refreshedItems.first(); 0709 QCOMPARE(entry.first.url().toLocalFile(), newPath); 0710 QCOMPARE(entry.second.url().toLocalFile(), path); 0711 disconnect(&m_dirLister, nullptr, this, nullptr); 0712 0713 // Let's see what KDirLister has in cache now 0714 KFileItem cachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(path)); 0715 QCOMPARE(cachedItem.url().toLocalFile(), path); 0716 KFileItem oldCachedItem = m_dirLister.findByUrl(QUrl::fromLocalFile(newPath)); 0717 QVERIFY(oldCachedItem.isNull()); 0718 m_refreshedItems.clear(); 0719 } 0720 0721 void KDirListerTest::testConcurrentListing() 0722 { 0723 const int origItemCount = m_items.count(); 0724 QCOMPARE(fileCount(), origItemCount); 0725 m_items.clear(); 0726 m_items2.clear(); 0727 0728 MyDirLister dirLister2; 0729 0730 const QString path = tempPath(); 0731 0732 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0733 connect(&dirLister2, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems2); 0734 0735 // Before dirLister2 has time to emit the items, let's make m_dirLister move to another dir. 0736 // This reproduces the use case "clicking on a folder in dolphin iconview, and dirlister2 0737 // is the one used by the "folder panel". m_dirLister is going to list the subdir, 0738 // while dirLister2 wants to list the folder that m_dirLister has just left. 0739 dirLister2.stop(); // like dolphin does, noop. 0740 dirLister2.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0741 m_dirLister.openUrl(QUrl::fromLocalFile(path + "subdir"), KDirLister::NoFlags); 0742 0743 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0744 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0745 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 0746 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0747 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0748 QCOMPARE(m_dirLister.spyClear.count(), 1); 0749 0750 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0751 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0752 #endif 0753 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0754 0755 QCOMPARE(m_items.count(), 0); 0756 0757 QCOMPARE(dirLister2.spyStarted.count(), 1); 0758 QCOMPARE(dirLister2.spyCompleted.count(), 0); 0759 QCOMPARE(dirLister2.spyCompletedQUrl.count(), 0); 0760 QCOMPARE(dirLister2.spyCanceled.count(), 0); 0761 QCOMPARE(dirLister2.spyCanceledQUrl.count(), 0); 0762 QCOMPARE(dirLister2.spyClear.count(), 1); 0763 0764 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0765 QCOMPARE(dirLister2.spyClearQUrl.count(), 0); 0766 #endif 0767 QCOMPARE(dirLister2.spyClearDir.count(), 0); 0768 0769 QCOMPARE(m_items2.count(), 0); 0770 QVERIFY(!m_dirLister.isFinished()); 0771 QVERIFY(!dirLister2.isFinished()); 0772 0773 // then wait for completed 0774 qDebug("waiting for completed"); 0775 0776 // QCOMPARE(m_dirLister.spyStarted.count(), 1); // 2 when subdir is already in cache. 0777 QTRY_COMPARE(m_dirLister.spyCompleted.count(), 1); 0778 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0779 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0780 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0781 QCOMPARE(m_dirLister.spyClear.count(), 1); 0782 0783 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0784 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0785 #endif 0786 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0787 0788 QCOMPARE(m_items.count(), 3); 0789 0790 QTRY_COMPARE(dirLister2.spyStarted.count(), 1); 0791 QTRY_COMPARE(dirLister2.spyCompleted.count(), 1); 0792 QCOMPARE(dirLister2.spyCompletedQUrl.count(), 1); 0793 QCOMPARE(dirLister2.spyCanceled.count(), 0); 0794 QCOMPARE(dirLister2.spyCanceledQUrl.count(), 0); 0795 QCOMPARE(dirLister2.spyClear.count(), 1); 0796 0797 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0798 QCOMPARE(dirLister2.spyClearQUrl.count(), 0); 0799 #endif 0800 QCOMPARE(dirLister2.spyClearDir.count(), 0); 0801 0802 QCOMPARE(m_items2.count(), origItemCount); 0803 if (!m_dirLister.isFinished()) { // false when an update is running because subdir is already in cache 0804 // TODO check why this fails QVERIFY(m_dirLister.spyCanceled.wait(1000)); 0805 QTest::qWait(1000); 0806 } 0807 0808 disconnect(&m_dirLister, nullptr, this, nullptr); 0809 disconnect(&dirLister2, nullptr, this, nullptr); 0810 } 0811 0812 void KDirListerTest::testConcurrentHoldingListing() 0813 { 0814 // #167851. 0815 // A dirlister holding the items, and a second dirlister does 0816 // openUrl(reload) (which triggers updateDirectory()) 0817 // and the first lister immediately does openUrl() (which emits cached items). 0818 0819 testOpenUrl(); // ensure m_dirLister holds the items. 0820 const int origItemCount = m_items.count(); 0821 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0822 0823 m_items.clear(); 0824 m_items2.clear(); 0825 const QString path = tempPath(); 0826 MyDirLister dirLister2; 0827 connect(&dirLister2, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems2); 0828 0829 dirLister2.openUrl(QUrl::fromLocalFile(path), KDirLister::Reload); // will start a list job 0830 QCOMPARE(dirLister2.spyStarted.count(), 1); 0831 QCOMPARE(dirLister2.spyCompleted.count(), 0); 0832 QCOMPARE(m_items.count(), 0); 0833 QCOMPARE(m_items2.count(), 0); 0834 0835 qDebug("calling m_dirLister.openUrl"); 0836 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); // should emit cached items, and then "join" the running listjob 0837 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0838 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0839 QCOMPARE(m_items.count(), 0); 0840 QCOMPARE(m_items2.count(), 0); 0841 0842 qDebug("waiting for completed"); 0843 QTRY_COMPARE(dirLister2.spyStarted.count(), 1); 0844 QTRY_COMPARE(dirLister2.spyCompleted.count(), 1); 0845 QCOMPARE(dirLister2.spyCompletedQUrl.count(), 1); 0846 QCOMPARE(dirLister2.spyCanceled.count(), 0); 0847 QCOMPARE(dirLister2.spyCanceledQUrl.count(), 0); 0848 QCOMPARE(dirLister2.spyClear.count(), 1); 0849 0850 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0851 QCOMPARE(dirLister2.spyClearQUrl.count(), 0); 0852 #endif 0853 QCOMPARE(dirLister2.spyClearDir.count(), 0); 0854 0855 QCOMPARE(m_items2.count(), origItemCount); 0856 0857 QTRY_COMPARE(m_dirLister.spyStarted.count(), 1); 0858 QTRY_COMPARE(m_dirLister.spyCompleted.count(), 1); 0859 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0860 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0861 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0862 QCOMPARE(m_dirLister.spyClear.count(), 1); 0863 0864 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0865 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0866 #endif 0867 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0868 0869 QVERIFY(dirLister2.isFinished()); 0870 QVERIFY(m_dirLister.isFinished()); 0871 disconnect(&m_dirLister, nullptr, this, nullptr); 0872 QCOMPARE(m_items.count(), origItemCount); 0873 } 0874 0875 void KDirListerTest::testConcurrentListingAndStop() 0876 { 0877 m_items.clear(); 0878 m_items2.clear(); 0879 0880 MyDirLister dirLister2; 0881 0882 // Use a new tempdir for this test, so that we don't use the cache at all. 0883 QTemporaryDir tempDir(homeTmpDir()); 0884 const QString path = tempDir.path() + '/'; 0885 createTestFile(path + "file_1"); 0886 createTestFile(path + "file_2"); 0887 createTestFile(path + "file_3"); 0888 0889 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0890 connect(&dirLister2, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems2); 0891 0892 // Before m_dirLister has time to emit the items, let's make dirLister2 call stop(). 0893 // This should not stop the list job for m_dirLister (#267709). 0894 dirLister2.openUrl(QUrl::fromLocalFile(path), KDirLister::Reload); 0895 m_dirLister.openUrl(QUrl::fromLocalFile(path) /*, KDirLister::Reload*/); 0896 0897 QCOMPARE(m_dirLister.spyStarted.count(), 1); 0898 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 0899 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 0900 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0901 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0902 QCOMPARE(m_dirLister.spyClear.count(), 1); 0903 0904 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0905 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0906 #endif 0907 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0908 0909 QCOMPARE(m_items.count(), 0); 0910 0911 QCOMPARE(dirLister2.spyStarted.count(), 1); 0912 QCOMPARE(dirLister2.spyCompleted.count(), 0); 0913 QCOMPARE(dirLister2.spyCompletedQUrl.count(), 0); 0914 QCOMPARE(dirLister2.spyCanceled.count(), 0); 0915 QCOMPARE(dirLister2.spyCanceledQUrl.count(), 0); 0916 QCOMPARE(dirLister2.spyClear.count(), 1); 0917 0918 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0919 QCOMPARE(dirLister2.spyClearQUrl.count(), 0); 0920 #endif 0921 QCOMPARE(dirLister2.spyClearDir.count(), 0); 0922 0923 QCOMPARE(m_items2.count(), 0); 0924 QVERIFY(!m_dirLister.isFinished()); 0925 QVERIFY(!dirLister2.isFinished()); 0926 0927 dirLister2.stop(); 0928 0929 QCOMPARE(dirLister2.spyStarted.count(), 1); 0930 QCOMPARE(dirLister2.spyCompleted.count(), 0); 0931 QCOMPARE(dirLister2.spyCompletedQUrl.count(), 0); 0932 QCOMPARE(dirLister2.spyCanceled.count(), 1); 0933 QCOMPARE(dirLister2.spyCanceledQUrl.count(), 1); 0934 QCOMPARE(dirLister2.spyClear.count(), 1); 0935 0936 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0937 QCOMPARE(dirLister2.spyClearQUrl.count(), 0); 0938 #endif 0939 QCOMPARE(dirLister2.spyClearDir.count(), 0); 0940 0941 QCOMPARE(m_items2.count(), 0); 0942 0943 // then wait for completed 0944 qDebug("waiting for completed"); 0945 QTRY_COMPARE(m_items.count(), 3); 0946 QTRY_COMPARE(m_items2.count(), 0); 0947 QTRY_VERIFY(m_dirLister.isFinished()); 0948 0949 // QCOMPARE(m_dirLister.spyStarted.count(), 1); // 2 when in cache 0950 QCOMPARE(m_dirLister.spyCompleted.count(), 1); 0951 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 1); 0952 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 0953 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 0954 QCOMPARE(m_dirLister.spyClear.count(), 1); 0955 0956 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 0957 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 0958 #endif 0959 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 0960 0961 disconnect(&m_dirLister, nullptr, this, nullptr); 0962 } 0963 0964 void KDirListerTest::testDeleteListerEarly() 0965 { 0966 // Do the same again, it should behave the same, even with the items in the cache 0967 testOpenUrl(); 0968 0969 // Start a second lister, it will get a cached items job, but delete it before the job can run 0970 // qDebug() << "=========================================="; 0971 { 0972 m_items.clear(); 0973 const QString path = tempPath(); 0974 MyDirLister secondDirLister; 0975 secondDirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 0976 QVERIFY(!secondDirLister.isFinished()); 0977 } 0978 // qDebug() << "=========================================="; 0979 0980 // Check if we didn't keep the deleted dirlister in one of our lists. 0981 // I guess the best way to do that is to just list the same dir again. 0982 testOpenUrl(); 0983 } 0984 0985 void KDirListerTest::testOpenUrlTwice() 0986 { 0987 // Calling openUrl(reload)+openUrl(normal) before listing even starts. 0988 const int origItemCount = m_items.count(); 0989 m_items.clear(); 0990 const QString path = tempPath(); 0991 MyDirLister secondDirLister; 0992 connect(&secondDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 0993 0994 secondDirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::Reload); // will start 0995 QCOMPARE(secondDirLister.spyStarted.count(), 1); 0996 QCOMPARE(secondDirLister.spyCompleted.count(), 0); 0997 0998 qDebug("calling openUrl again"); 0999 secondDirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); // will stop + start 1000 1001 qDebug("waiting for completed"); 1002 QTRY_COMPARE(secondDirLister.spyStarted.count(), 2); 1003 QTRY_COMPARE(secondDirLister.spyCompleted.count(), 1); 1004 QCOMPARE(secondDirLister.spyCompletedQUrl.count(), 1); 1005 QCOMPARE(secondDirLister.spyCanceled.count(), 0); // should not be emitted, see next test 1006 QCOMPARE(secondDirLister.spyCanceledQUrl.count(), 0); 1007 QCOMPARE(secondDirLister.spyClear.count(), 2); 1008 1009 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1010 QCOMPARE(secondDirLister.spyClearQUrl.count(), 0); 1011 #endif 1012 QCOMPARE(secondDirLister.spyClearDir.count(), 0); 1013 1014 if (origItemCount) { // 0 if running this test separately 1015 QCOMPARE(m_items.count(), origItemCount); 1016 } 1017 QVERIFY(secondDirLister.isFinished()); 1018 disconnect(&secondDirLister, nullptr, this, nullptr); 1019 } 1020 1021 void KDirListerTest::testOpenUrlTwiceWithKeep() 1022 { 1023 // Calling openUrl(reload)+openUrl(keep) on a new dir, 1024 // before listing even starts (#177387) 1025 // Well, in 177387 the second openUrl call was made from within slotCanceled 1026 // called by the first openUrl 1027 // (slotLoadingFinished -> setCurrentItem -> expandToUrl -> listDir), 1028 // which messed things up in kdirlister (unexpected reentrancy). 1029 m_items.clear(); 1030 const QString path = tempPath() + "newsubdir"; 1031 QDir().mkdir(path); 1032 MyDirLister secondDirLister; 1033 connect(&secondDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1034 1035 secondDirLister.openUrl(QUrl::fromLocalFile(path)); // will start a list job 1036 QCOMPARE(secondDirLister.spyStarted.count(), 1); 1037 QCOMPARE(secondDirLister.spyCanceled.count(), 0); 1038 QCOMPARE(secondDirLister.spyCompleted.count(), 0); 1039 1040 qDebug("calling openUrl again"); 1041 secondDirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::Keep); // stops and restarts the job 1042 1043 qDebug("waiting for completed"); 1044 QTRY_COMPARE(secondDirLister.spyStarted.count(), 2); 1045 QTRY_COMPARE(secondDirLister.spyCompleted.count(), 1); 1046 QCOMPARE(secondDirLister.spyCompletedQUrl.count(), 1); 1047 QCOMPARE(secondDirLister.spyCanceled.count(), 0); // should not be emitted, it led to recursion 1048 QCOMPARE(secondDirLister.spyCanceledQUrl.count(), 0); 1049 QCOMPARE(secondDirLister.spyClear.count(), 1); 1050 1051 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1052 QCOMPARE(secondDirLister.spyClearQUrl.count(), 1); 1053 #endif 1054 QCOMPARE(secondDirLister.spyClearDir.count(), 1); 1055 1056 QCOMPARE(m_items.count(), 0); 1057 QVERIFY(secondDirLister.isFinished()); 1058 disconnect(&secondDirLister, nullptr, this, nullptr); 1059 1060 QDir().remove(path); 1061 } 1062 1063 void KDirListerTest::testOpenAndStop() 1064 { 1065 m_items.clear(); 1066 const QString path = QStringLiteral("/"); // better not use a directory that we already listed! 1067 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1068 1069 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 1070 qDebug() << "Calling stop!"; 1071 m_dirLister.stop(); // we should also test stop(QUrl::fromLocalFile(path))... 1072 1073 QCOMPARE(m_dirLister.spyStarted.count(), 1); // The call to openUrl itself, emits started 1074 QCOMPARE(m_dirLister.spyCompleted.count(), 0); // we had time to stop before the job even started 1075 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 1076 QCOMPARE(m_dirLister.spyCanceled.count(), 1); 1077 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 1); 1078 QCOMPARE(m_dirLister.spyClear.count(), 1); 1079 1080 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1081 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 1082 #endif 1083 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 1084 1085 QCOMPARE(m_items.count(), 0); // we had time to stop before the job even started 1086 QVERIFY(m_dirLister.isFinished()); 1087 disconnect(&m_dirLister, nullptr, this, nullptr); 1088 } 1089 1090 // A bug in the decAutoUpdate/incAutoUpdate logic made KDirLister stop watching a directory for changes, 1091 // and never watch it again when opening it from the cache. 1092 void KDirListerTest::testBug211472() 1093 { 1094 m_items.clear(); 1095 1096 QTemporaryDir newDir(homeTmpDir()); 1097 const QString path = newDir.path() + "/newsubdir/"; 1098 QDir().mkdir(path); 1099 MyDirLister dirLister; 1100 connect(&dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1101 1102 dirLister.openUrl(QUrl::fromLocalFile(path)); 1103 QVERIFY(dirLister.spyCompleted.wait(1000)); 1104 QVERIFY(dirLister.isFinished()); 1105 QVERIFY(m_items.isEmpty()); 1106 1107 // This block is required to trigger bug 211472. 1108 1109 // Go 'up' to the parent of 'newsubdir'. 1110 dirLister.openUrl(QUrl::fromLocalFile(newDir.path())); 1111 QVERIFY(dirLister.spyCompleted.wait(1000)); 1112 QTRY_VERIFY(dirLister.isFinished()); 1113 QTRY_VERIFY(!m_items.isEmpty()); 1114 m_items.clear(); 1115 1116 // Create a file in 'newsubdir' while we are listing its parent dir. 1117 createTestFile(path + "newFile-1"); 1118 // At this point, newsubdir is not used, so it's moved to the cache. 1119 // This happens in checkUpdate, called when receiving a notification for the cached dir, 1120 // this is why this unittest needs to create a test file in the subdir. 1121 1122 // wait a second and ensure the list is still empty afterwards 1123 QTest::qWait(1000); 1124 QTRY_VERIFY(m_items.isEmpty()); 1125 1126 // Return to 'newsubdir'. It will be emitted from the cache, then an update will happen. 1127 dirLister.openUrl(QUrl::fromLocalFile(path)); 1128 // Check that completed is emitted twice 1129 QVERIFY(dirLister.spyCompleted.wait(1000)); 1130 QVERIFY(dirLister.spyCompleted.wait(1000)); 1131 QTRY_VERIFY(dirLister.isFinished()); 1132 QTRY_COMPARE(m_items.count(), 1); 1133 m_items.clear(); 1134 1135 // Now try to create a second file in 'newsubdir' and verify that the 1136 // dir lister notices it. 1137 QTest::qWait(1000); // We need a 1s timestamp difference on the dir, otherwise FAM won't notice anything. 1138 1139 createTestFile(path + "newFile-2"); 1140 QTRY_COMPARE(m_items.count(), 1); 1141 1142 newDir.remove(); 1143 QSignalSpy spyClear(&dirLister, qOverload<>(&KCoreDirLister::clear)); 1144 QVERIFY(spyClear.wait(1000)); 1145 } 1146 1147 void KDirListerTest::testRenameCurrentDir() // #294445 1148 { 1149 m_items.clear(); 1150 1151 const QString path = tempPath() + "newsubdir-1"; 1152 QVERIFY(QDir().mkdir(path)); 1153 MyDirLister secondDirLister; 1154 connect(&secondDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1155 1156 secondDirLister.openUrl(QUrl::fromLocalFile(path)); 1157 QSignalSpy spyCompleted(&secondDirLister, qOverload<>(&KCoreDirLister::completed)); 1158 QVERIFY(spyCompleted.wait(1000)); 1159 QVERIFY(secondDirLister.isFinished()); 1160 QVERIFY(m_items.empty()); 1161 QCOMPARE(secondDirLister.rootItem().url().toLocalFile(), path); 1162 1163 const QString newPath = tempPath() + "newsubdir-2"; 1164 QVERIFY(QDir().rename(path, newPath)); 1165 org::kde::KDirNotify::emitFileRenamed(QUrl::fromLocalFile(path), QUrl::fromLocalFile(newPath)); 1166 QSignalSpy spyRedirection(&secondDirLister, qOverload<const QUrl &, const QUrl &>(&KCoreDirLister::redirection)); 1167 QVERIFY(spyRedirection.wait(1000)); 1168 1169 // Check that the URL of the root item got updated 1170 QCOMPARE(secondDirLister.rootItem().url().toLocalFile(), newPath); 1171 1172 disconnect(&secondDirLister, nullptr, this, nullptr); 1173 QDir().rmdir(newPath); 1174 } 1175 1176 void KDirListerTest::slotOpenUrlOnRename(const QUrl &newUrl) 1177 { 1178 QVERIFY(m_dirLister.openUrl(newUrl)); 1179 } 1180 1181 // This tests for a crash if you connect redirects to openUrl, due 1182 // to internal data being inconsistently exposed. 1183 // Matches usage in gwenview. 1184 void KDirListerTest::testRenameCurrentDirOpenUrl() 1185 { 1186 m_items.clear(); 1187 const QString path = tempPath() + "newsubdir-1/"; 1188 QVERIFY(QDir().mkdir(path)); 1189 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1190 1191 m_dirLister.openUrl(QUrl::fromLocalFile(path)); 1192 QSignalSpy spyCompleted(&m_dirLister, qOverload<>(&KCoreDirLister::completed)); 1193 // Wait for the signal completed to be emitted 1194 QVERIFY(spyCompleted.wait(1000)); 1195 QVERIFY(m_dirLister.isFinished()); 1196 1197 const QString newPath = tempPath() + "newsubdir-2"; 1198 QVERIFY(QDir().rename(path, newPath)); 1199 1200 org::kde::KDirNotify::emitFileRenamed(QUrl::fromLocalFile(path), QUrl::fromLocalFile(newPath)); 1201 1202 // Connect the redirection to openURL, so that on a rename the new location is opened. 1203 // This matches usage in gwenview, and crashes 1204 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 80) 1205 connect(&m_dirLister, qOverload<const QUrl &>(&KCoreDirLister::redirection), this, &KDirListerTest::slotOpenUrlOnRename); 1206 #else 1207 connect(&m_dirLister, qOverload<const QUrl &, const QUrl &>(&KCoreDirLister::redirection), this, [this](const QUrl &, const QUrl &newUrl) { 1208 slotOpenUrlOnRename(newUrl); 1209 }); 1210 #endif 1211 1212 QTRY_VERIFY(m_dirLister.isFinished()); 1213 disconnect(&m_dirLister, nullptr, this, nullptr); 1214 QDir().rmdir(newPath); 1215 } 1216 1217 void KDirListerTest::testRedirection() 1218 { 1219 m_items.clear(); 1220 const QUrl url(QStringLiteral("file://somemachine/")); 1221 1222 if (!KProtocolInfo::isKnownProtocol(QStringLiteral("smb"))) { 1223 QSKIP("smb not installed"); 1224 } 1225 1226 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1227 // The call to openUrl itself, emits started 1228 m_dirLister.openUrl(url, KDirLister::NoFlags); 1229 1230 QCOMPARE(m_dirLister.spyStarted.count(), 1); 1231 QCOMPARE(m_dirLister.spyCompleted.count(), 0); 1232 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 1233 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 1234 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 1235 QCOMPARE(m_dirLister.spyClear.count(), 1); 1236 1237 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1238 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 1239 #endif 1240 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 1241 1242 QCOMPARE(m_dirLister.spyRedirection.count(), 0); 1243 QCOMPARE(m_items.count(), 0); 1244 QVERIFY(!m_dirLister.isFinished()); 1245 1246 // then wait for the redirection signal 1247 qDebug("waiting for redirection"); 1248 QTRY_COMPARE(m_dirLister.spyStarted.count(), 1); 1249 QCOMPARE(m_dirLister.spyCompleted.count(), 0); // we stopped before the listing. 1250 QCOMPARE(m_dirLister.spyCompletedQUrl.count(), 0); 1251 QCOMPARE(m_dirLister.spyCanceled.count(), 0); 1252 QCOMPARE(m_dirLister.spyCanceledQUrl.count(), 0); 1253 QTRY_COMPARE(m_dirLister.spyClear.count(), 2); // redirection cleared a second time (just in case...) 1254 1255 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1256 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 1257 #endif 1258 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 1259 1260 QTRY_COMPARE(m_dirLister.spyRedirection.count(), 1); 1261 QVERIFY(m_items.isEmpty()); 1262 QVERIFY(!m_dirLister.isFinished()); 1263 1264 m_dirLister.stop(url); 1265 QVERIFY(!m_dirLister.isFinished()); 1266 disconnect(&m_dirLister, nullptr, this, nullptr); 1267 } 1268 1269 void KDirListerTest::testListEmptyDirFromCache() // #278431 1270 { 1271 m_items.clear(); 1272 1273 QTemporaryDir newDir(homeTmpDir()); 1274 const QUrl url = QUrl::fromLocalFile(newDir.path()); 1275 1276 // List and watch an empty dir 1277 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1278 m_dirLister.openUrl(url); 1279 QSignalSpy spyCompleted(&m_dirLister, qOverload<>(&KCoreDirLister::completed)); 1280 QVERIFY(spyCompleted.wait(1000)); 1281 QVERIFY(m_dirLister.isFinished()); 1282 QVERIFY(m_items.isEmpty()); 1283 1284 // List it with two more dirlisters (one will create a cached items job, the second should also benefit from it) 1285 MyDirLister secondDirLister; 1286 connect(&secondDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1287 secondDirLister.openUrl(url); 1288 MyDirLister thirdDirLister; 1289 connect(&thirdDirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1290 thirdDirLister.openUrl(url); 1291 1292 // The point of this test is that (with DEBUG_CACHE enabled) it used to assert here 1293 // with "HUH? Lister KDirLister(0x7ffd1f044260) is supposed to be listing, but has no job!" 1294 // due to the if (!itemU->lstItems.isEmpty()) check which is now removed. 1295 1296 QVERIFY(!secondDirLister.isFinished()); // we didn't go to the event loop yet 1297 QSignalSpy spySecondCompleted(&secondDirLister, qOverload<>(&KCoreDirLister::completed)); 1298 QVERIFY(spySecondCompleted.wait(1000)); 1299 if (!thirdDirLister.isFinished()) { 1300 QSignalSpy spyThirdCompleted(&thirdDirLister, qOverload<>(&KCoreDirLister::completed)); 1301 QVERIFY(spyThirdCompleted.wait(1000)); 1302 } 1303 } 1304 1305 void KDirListerTest::testWatchingAfterCopyJob() // #331582 1306 { 1307 m_items.clear(); 1308 1309 QTemporaryDir newDir(homeTmpDir()); 1310 const QString path = newDir.path() + '/'; 1311 1312 // List and watch an empty dir 1313 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1314 m_dirLister.openUrl(QUrl::fromLocalFile(path)); 1315 QSignalSpy spyCompleted(&m_dirLister, qOverload<>(&KCoreDirLister::completed)); 1316 QVERIFY(spyCompleted.wait(1000)); 1317 QVERIFY(m_dirLister.isFinished()); 1318 QVERIFY(m_items.isEmpty()); 1319 1320 // Create three subfolders. 1321 QVERIFY(QDir().mkdir(path + "New Folder")); 1322 QVERIFY(QDir().mkdir(path + "New Folder 1")); 1323 QVERIFY(QDir().mkdir(path + "New Folder 2")); 1324 1325 QVERIFY(spyCompleted.wait(1000)); 1326 QTRY_VERIFY(m_dirLister.isFinished()); 1327 QTRY_COMPARE(m_items.count(), 3); 1328 1329 // Create a new file and verify that the dir lister notices it. 1330 m_items.clear(); 1331 createTestFile(path + QLatin1Char('a')); 1332 QVERIFY(spyCompleted.wait(1000)); 1333 QTRY_VERIFY(m_dirLister.isFinished()); 1334 QTRY_COMPARE(m_items.count(), 1); 1335 1336 // Rename one of the subfolders. 1337 const QString oldPath = path + "New Folder 1"; 1338 const QString newPath = path + "New Folder 1a"; 1339 1340 // NOTE: The following two lines are required to trigger the bug! 1341 KIO::Job *job = KIO::moveAs(QUrl::fromLocalFile(oldPath), QUrl::fromLocalFile(newPath), KIO::HideProgressInfo); 1342 job->exec(); 1343 1344 // Now try to create a second new file and verify that the 1345 // dir lister notices it. 1346 m_items.clear(); 1347 createTestFile(path + QLatin1Char('b')); 1348 1349 // This should end up in "KCoreDirListerCache::slotFileDirty" 1350 QTRY_COMPARE(m_items.count(), 1); 1351 1352 newDir.remove(); 1353 QSignalSpy clearSpy(&m_dirLister, qOverload<>(&KCoreDirLister::clear)); 1354 QVERIFY(clearSpy.wait(1000)); 1355 } 1356 1357 void KDirListerTest::testRemoveWatchedDirectory() 1358 { 1359 m_items.clear(); 1360 1361 QTemporaryDir newDir(homeTmpDir()); 1362 const QString path = newDir.path() + '/'; 1363 1364 // List and watch an empty dir 1365 connect(&m_dirLister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1366 m_dirLister.openUrl(QUrl::fromLocalFile(path)); 1367 QSignalSpy spyCompleted(&m_dirLister, qOverload<>(&KCoreDirLister::completed)); 1368 QVERIFY(spyCompleted.wait(1000)); 1369 QTRY_VERIFY(m_dirLister.isFinished()); 1370 QTRY_VERIFY(m_items.isEmpty()); 1371 1372 // Create a subfolder. 1373 const QString subDirPath = path + "abc"; 1374 QVERIFY(QDir().mkdir(subDirPath)); 1375 1376 QVERIFY(spyCompleted.wait(1000)); 1377 QTRY_VERIFY(m_dirLister.isFinished()); 1378 QTRY_COMPARE(m_items.count(), 1); 1379 const KFileItem item = m_items.at(0); 1380 1381 // Watch the subfolder for changes, independently. 1382 // This is what triggers the bug. 1383 // (Technically, this could become a KDirWatch unittest, but if one day we use QFSW, good to have the tests here) 1384 KDirWatch watcher; 1385 watcher.addDir(subDirPath); 1386 1387 // Remove the subfolder. 1388 m_items.clear(); 1389 QVERIFY(QDir().rmdir(path + "abc")); 1390 1391 // This should trigger an update. 1392 QVERIFY(spyCompleted.wait(1000)); 1393 QVERIFY(m_dirLister.isFinished()); 1394 QCOMPARE(m_items.count(), 0); 1395 QCOMPARE(m_dirLister.spyItemsDeleted.count(), 1); 1396 const KFileItem deletedItem = m_dirLister.spyItemsDeleted.at(0).at(0).value<KFileItemList>().at(0); 1397 QCOMPARE(item, deletedItem); 1398 } 1399 1400 void KDirListerTest::testDirPermissionChange() 1401 { 1402 QTemporaryDir tempDir(homeTmpDir()); 1403 1404 const QString path = tempDir.path() + '/'; 1405 const QString subdir = path + QLatin1String("subdir"); 1406 QVERIFY(QDir().mkdir(subdir)); 1407 1408 MyDirLister mylister; 1409 mylister.openUrl(QUrl::fromLocalFile(tempDir.path())); 1410 QSignalSpy spyCompleted(&mylister, qOverload<>(&KCoreDirLister::completed)); 1411 QVERIFY(spyCompleted.wait(1000)); 1412 1413 KFileItemList list = mylister.items(); 1414 QVERIFY(mylister.isFinished()); 1415 QCOMPARE(list.count(), 1); 1416 QCOMPARE(mylister.rootItem().url().toLocalFile(), tempDir.path()); 1417 1418 const mode_t permissions = (S_IRUSR | S_IWUSR | S_IXUSR); 1419 KIO::SimpleJob *job = KIO::chmod(list.first().url(), permissions); 1420 QVERIFY(job->exec()); 1421 1422 QSignalSpy spyRefreshItems(&mylister, &KCoreDirLister::refreshItems); 1423 QVERIFY(spyRefreshItems.wait(2000)); 1424 1425 list = mylister.items(); 1426 QCOMPARE(list.first().permissions(), permissions); 1427 QVERIFY(QDir().rmdir(subdir)); 1428 } 1429 1430 void KDirListerTest::slotNewItems(const KFileItemList &lst) 1431 { 1432 m_items += lst; 1433 } 1434 1435 void KDirListerTest::slotNewItems2(const KFileItemList &lst) 1436 { 1437 m_items2 += lst; 1438 } 1439 1440 void KDirListerTest::slotRefreshItems(const QList<QPair<KFileItem, KFileItem>> &lst) 1441 { 1442 m_refreshedItems += lst; 1443 Q_EMIT refreshItemsReceived(); 1444 } 1445 1446 void KDirListerTest::slotRefreshItems2(const QList<QPair<KFileItem, KFileItem>> &lst) 1447 { 1448 m_refreshedItems2 += lst; 1449 } 1450 1451 void KDirListerTest::testCopyAfterListingAndMove() // #353195 1452 { 1453 const QString dirA = tempPath() + "a"; 1454 QVERIFY(QDir().mkdir(dirA)); 1455 const QString dirB = tempPath() + "b"; 1456 QVERIFY(QDir().mkdir(dirB)); 1457 1458 // ensure m_dirLister holds the items. 1459 m_dirLister.openUrl(QUrl::fromLocalFile(tempPath()), KDirLister::NoFlags); 1460 QSignalSpy spyCompleted(&m_dirLister, qOverload<>(&KCoreDirLister::completed)); 1461 QVERIFY(spyCompleted.wait()); 1462 1463 // Move b into a 1464 KIO::Job *moveJob = KIO::move(QUrl::fromLocalFile(dirB), QUrl::fromLocalFile(dirA)); 1465 moveJob->setUiDelegate(nullptr); 1466 QVERIFY(moveJob->exec()); 1467 QVERIFY(QFileInfo(tempPath() + "a/b").isDir()); 1468 1469 // Give some time to processPendingUpdates 1470 QTest::qWait(1000); 1471 1472 // Copy folder a elsewhere 1473 const QString dest = tempPath() + "subdir"; 1474 KIO::Job *copyJob = KIO::copy(QUrl::fromLocalFile(dirA), QUrl::fromLocalFile(dest)); 1475 copyJob->setUiDelegate(nullptr); 1476 QVERIFY(copyJob->exec()); 1477 QVERIFY(QFileInfo(tempPath() + "subdir/a/b").isDir()); 1478 } 1479 1480 void KDirListerTest::testRenameDirectory() // #401552 1481 { 1482 // Create the directory structure to reproduce the bug in a reliable way 1483 const QString dirW = tempPath() + "w"; 1484 QVERIFY(QDir().mkdir(dirW)); 1485 const QString dirW1 = tempPath() + "w/Files"; 1486 QVERIFY(QDir().mkdir(dirW1)); 1487 const QString dirW2 = tempPath() + "w/Files/Files"; 1488 QVERIFY(QDir().mkdir(dirW2)); 1489 // Place some empty files in each directory 1490 for (int i = 0; i < 50; i++) { 1491 createSimpleFile(dirW + QString("t_%1").arg(i)); 1492 } 1493 for (int i = 0; i < 50; i++) { 1494 createSimpleFile(dirW + QString("z_%1").arg(i)); 1495 } 1496 // Place some empty files with prefix Files in w. Note that / is missing. 1497 for (int i = 0; i < 50; i++) { 1498 createSimpleFile(dirW1 + QString("t_%1").arg(i)); 1499 } 1500 for (int i = 0; i < 50; i++) { 1501 createSimpleFile(dirW1 + QString("z_%1").arg(i)); 1502 } 1503 // Place some empty files with prefix Files in w/Files. Note that / is missing. 1504 for (int i = 0; i < 50; i++) { 1505 createSimpleFile(dirW2 + QString("t_%1").arg(i)); 1506 } 1507 for (int i = 0; i < 50; i++) { 1508 createSimpleFile(dirW2 + QString("z_%1").arg(i)); 1509 } 1510 // Listen to the w directory 1511 m_dirLister.openUrl(QUrl::fromLocalFile(dirW), KDirLister::NoFlags); 1512 1513 // Try to reproduce the bug #401552 renaming the w directory several times if needed 1514 const QStringList dirs = {dirW + "___", dirW + QLatin1Char('_'), dirW + "______", dirW + "_c", dirW + "___", dirW + "_________"}; 1515 1516 QString currDir = dirW; 1517 KIO::SimpleJob *job = nullptr; 1518 // Connect the redirection to openURL, so that on a rename the new location is opened. 1519 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 80) 1520 connect(&m_dirLister, qOverload<const QUrl &>(&KCoreDirLister::redirection), this, &KDirListerTest::slotOpenUrlOnRename); 1521 #else 1522 connect(&m_dirLister, qOverload<const QUrl &, const QUrl &>(&KCoreDirLister::redirection), this, [this](const QUrl &, const QUrl &newUrl) { 1523 slotOpenUrlOnRename(newUrl); 1524 }); 1525 #endif 1526 1527 for (int i = 0; i < dirs.size(); i++) { 1528 // Wait for the listener to get all files 1529 QTRY_VERIFY(m_dirLister.isFinished()); 1530 // Do the rename 1531 QString newDir = dirs.at(i); 1532 job = KIO::rename(QUrl::fromLocalFile(currDir), QUrl::fromLocalFile(newDir), KIO::HideProgressInfo); 1533 QVERIFY2(job->exec(), qPrintable(job->errorString())); 1534 QTest::qWait(500); // Without the delay the crash doesn't happen 1535 currDir = newDir; 1536 } 1537 disconnect(&m_dirLister, nullptr, this, nullptr); 1538 } 1539 1540 void KDirListerTest::testRequestMimeType() 1541 { 1542 // Use a new tempdir and lister instance for this test, so that we don't use any cache at all. 1543 QTemporaryDir tempDir(homeTmpDir()); 1544 QString path = tempDir.path() + QLatin1Char('/'); 1545 1546 createTestFile(path + "/file_1"); 1547 createTestFile(path + "/file_2.txt"); 1548 createTestFile(path + "/file_3.cpp"); 1549 createTestFile(path + "/file_3.md"); 1550 1551 MyDirLister lister; 1552 // Explicitly set requestMimeTypeWhileListing to false so we know what state 1553 // it is in. 1554 lister.setRequestMimeTypeWhileListing(false); 1555 lister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 1556 1557 QTRY_VERIFY(lister.isFinished()); 1558 1559 auto items = lister.items(); 1560 for (auto item : std::as_const(items)) { 1561 QVERIFY(!item.isMimeTypeKnown()); 1562 } 1563 1564 // Verify that the mime types are what we expect them to be 1565 QCOMPARE(items[0].mimetype(), QStringLiteral("application/octet-stream")); 1566 QCOMPARE(items[1].mimetype(), QStringLiteral("text/plain")); 1567 QCOMPARE(items[2].mimetype(), QStringLiteral("text/x-c++src")); 1568 QCOMPARE(items[3].mimetype(), QStringLiteral("text/markdown")); 1569 1570 lister.setRequestMimeTypeWhileListing(true); 1571 lister.openUrl(QUrl::fromLocalFile(path), KDirLister::Reload); 1572 1573 QTRY_VERIFY(lister.isFinished()); 1574 1575 // If requestMimeTypeWhileListing is on, we should know the mime type of 1576 // items when they have been listed. 1577 items = lister.items(); 1578 for (auto item : std::as_const(items)) { 1579 QVERIFY(item.isMimeTypeKnown()); 1580 } 1581 1582 // Verify that the mime types are what we expect them to be 1583 QCOMPARE(items[0].mimetype(), QStringLiteral("application/octet-stream")); 1584 QCOMPARE(items[1].mimetype(), QStringLiteral("text/plain")); 1585 QCOMPARE(items[2].mimetype(), QStringLiteral("text/x-c++src")); 1586 QCOMPARE(items[3].mimetype(), QStringLiteral("text/markdown")); 1587 } 1588 1589 void KDirListerTest::testMimeFilter_data() 1590 { 1591 QTest::addColumn<QStringList>("files"); 1592 QTest::addColumn<QStringList>("mimeTypes"); 1593 QTest::addColumn<QStringList>("filteredFiles"); 1594 1595 const QStringList files = {"bla.txt", "main.cpp", "main.c", "image.jpeg"}; 1596 1597 QTest::newRow("single_file_exact_mimetype") << files << QStringList{"text/x-c++src"} << QStringList{"main.cpp"}; 1598 QTest::newRow("inherited_mimetype") << files << QStringList{"text/plain"} << QStringList{"bla.txt", "main.cpp", "main.c"}; 1599 QTest::newRow("no_match") << files << QStringList{"audio/flac"} << QStringList{}; 1600 } 1601 1602 void KDirListerTest::testMimeFilter() 1603 { 1604 // Use a new tempdir and lister instance for this test, so that we don't use any cache at all. 1605 QTemporaryDir tempDir(homeTmpDir()); 1606 QString path = tempDir.path() + '/'; 1607 1608 QFETCH(QStringList, files); 1609 QFETCH(QStringList, mimeTypes); 1610 QFETCH(QStringList, filteredFiles); 1611 1612 for (const QString &fileName : files) { 1613 createTestFile(path + fileName); 1614 } 1615 1616 MyDirLister lister; 1617 lister.setMimeFilter(mimeTypes); 1618 lister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 1619 1620 QSignalSpy spyCompleted(&lister, qOverload<>(&KCoreDirLister::completed)); 1621 QVERIFY(spyCompleted.wait(1000)); 1622 1623 QCOMPARE(lister.items().size(), filteredFiles.size()); 1624 1625 const auto items = lister.items(); 1626 for (const auto &item : items) { 1627 QVERIFY(filteredFiles.indexOf(item.name()) != -1); 1628 } 1629 } 1630 1631 void KDirListerTest::testDeleteCurrentDir() 1632 { 1633 // ensure m_dirLister holds the items. 1634 m_dirLister.openUrl(QUrl::fromLocalFile(tempPath()), KDirLister::NoFlags); 1635 m_dirLister.clearSpies(); 1636 KIO::DeleteJob *job = KIO::del(QUrl::fromLocalFile(tempPath()), KIO::HideProgressInfo); 1637 bool ok = job->exec(); 1638 QVERIFY(ok); 1639 QTRY_COMPARE(m_dirLister.spyClear.count(), 1); 1640 1641 #if KIOCORE_BUILD_DEPRECATED_SINCE(5, 79) 1642 QCOMPARE(m_dirLister.spyClearQUrl.count(), 0); 1643 #endif 1644 QCOMPARE(m_dirLister.spyClearDir.count(), 0); 1645 1646 QList<QUrl> deletedUrls; 1647 for (int i = 0; i < m_dirLister.spyItemsDeleted.count(); ++i) { 1648 deletedUrls += m_dirLister.spyItemsDeleted[i][0].value<KFileItemList>().urlList(); 1649 } 1650 // qDebug() << deletedUrls; 1651 QUrl currentDirUrl = QUrl::fromLocalFile(tempPath()).adjusted(QUrl::StripTrailingSlash); 1652 // Sometimes I get ("current/subdir", "current") here, but that seems ok. 1653 QVERIFY(deletedUrls.contains(currentDirUrl)); 1654 } 1655 1656 void KDirListerTest::testForgetDir() 1657 { 1658 QTemporaryDir tempDir(homeTmpDir()); 1659 QString path = tempDir.path(); 1660 createTestFile(path + "/file_1"); 1661 1662 m_dirLister.openUrl(QUrl::fromLocalFile(path), KDirLister::Keep); 1663 QVERIFY(m_dirLister.spyCompleted.wait()); 1664 1665 m_dirLister.forgetDirs(QUrl::fromLocalFile(path)); 1666 1667 QSignalSpy addedSpy(&m_dirLister, &MyDirLister::itemsAdded); 1668 createTestFile(path + "/file_2"); 1669 QVERIFY(!addedSpy.wait(1000)); // to allow for KDirWatch's internal 500ms timer 1670 } 1671 1672 int KDirListerTest::fileCount() const 1673 { 1674 return QDir(tempPath()).entryList(QDir::AllEntries | QDir::NoDotAndDotDot).count(); 1675 } 1676 1677 void KDirListerTest::createSimpleFile(const QString &fileName) 1678 { 1679 QFile file(fileName); 1680 QVERIFY(file.open(QIODevice::WriteOnly)); 1681 file.write(QByteArray("foo")); 1682 file.close(); 1683 } 1684 1685 void KDirListerTest::fillDirLister2(MyDirLister &lister, const QString &path) 1686 { 1687 m_items2.clear(); 1688 connect(&lister, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems2); 1689 connect(&m_dirLister, &KCoreDirLister::refreshItems, this, &KDirListerTest::slotRefreshItems2); 1690 lister.openUrl(QUrl::fromLocalFile(path), KDirLister::NoFlags); 1691 QTRY_VERIFY(lister.isFinished()); 1692 } 1693 1694 void KDirListerTest::waitUntilMTimeChange(const QString &path) 1695 { 1696 // Wait until the current second is more than the file's mtime 1697 // otherwise this change will go unnoticed 1698 1699 QFileInfo fi(path); 1700 QVERIFY(fi.exists()); 1701 const QDateTime mtime = fi.lastModified(); 1702 waitUntilAfter(mtime); 1703 } 1704 1705 void KDirListerTest::waitUntilAfter(const QDateTime &ctime) 1706 { 1707 int totalWait = 0; 1708 QDateTime now; 1709 Q_FOREVER { 1710 now = QDateTime::currentDateTime(); 1711 if (now.toSecsSinceEpoch() == ctime.toSecsSinceEpoch()) { // truncate milliseconds 1712 totalWait += 50; 1713 QTest::qWait(50); 1714 } else { 1715 QVERIFY(now > ctime); // can't go back in time ;) 1716 QTest::qWait(50); // be safe 1717 break; 1718 } 1719 } 1720 // if (totalWait > 0) 1721 qDebug() << "Waited" << totalWait << "ms so that now" << now.toString(Qt::ISODate) << "is >" << ctime.toString(Qt::ISODate); 1722 } 1723 1724 // A bug in the decAutoUpdate/incAutoUpdate logic made KDirLister stop watching a directory for changes, 1725 // and stop watching a directory because a separate lister left a directory open in another lister 1726 void KDirListerTest::testBug386763() 1727 { 1728 QTemporaryDir newDir(homeTmpDir()); 1729 const QString path = newDir.path() + "/newsubdir/"; 1730 const QString otherpath = newDir.path() + "/othersubdir/"; 1731 1732 QDir().mkdir(path); 1733 MyDirLister dirLister; 1734 dirLister.openUrl(QUrl::fromLocalFile(path)); 1735 1736 // second lister opening same dir 1737 MyDirLister dirLister2; 1738 dirLister2.openUrl(QUrl::fromLocalFile(path)); 1739 QCOMPARE(dirLister2.spyCompleted.count(), 0); 1740 1741 connect(&dirLister2, &KCoreDirLister::newItems, this, &KDirListerTest::slotNewItems); 1742 QVERIFY(dirLister.spyCompleted.wait(500)); 1743 QVERIFY(dirLister.isFinished()); 1744 QVERIFY(m_items.isEmpty()); 1745 1746 // first lister opening another dir 1747 dirLister.openUrl(QUrl::fromLocalFile(otherpath)); 1748 1749 // Create a file in 'newsubdir' while still opened in dirLister2 1750 // bug was that the watch on ’newsubdir’ when dirLister left this dir 1751 // eventhough dirLister2 is still listing it 1752 QCOMPARE(dirLister2.spyCompleted.count(), 1); 1753 createTestFile(path + "newFile-1"); 1754 1755 QTRY_COMPARE(m_items.count(), 1); 1756 QVERIFY(KDirWatch::self()->contains(path)); 1757 1758 dirLister2.openUrl(QUrl::fromLocalFile(otherpath)); 1759 // checks we still watch the old path when second lister leaves it as it should be now in cache 1760 QVERIFY(KDirWatch::self()->contains(path)); 1761 1762 newDir.remove(); 1763 } 1764 1765 void KDirListerTest::testCacheEviction() 1766 { 1767 QTemporaryDir newDir(homeTmpDir()); 1768 1769 MyDirLister dirLister; 1770 dirLister.openUrl(QUrl::fromLocalFile(newDir.path())); 1771 QVERIFY(dirLister.spyCompleted.wait(500)); 1772 QVERIFY(dirLister.isFinished()); 1773 QVERIFY(KDirWatch::self()->contains(newDir.path())); 1774 1775 for (int i = 0; i < 12; i++) { 1776 const QString newDirPath = newDir.path() + QString("dir_%1").arg(i); 1777 QVERIFY(QDir().mkdir(newDirPath)); 1778 1779 dirLister.openUrl(QUrl::fromLocalFile(newDirPath)); 1780 QVERIFY(dirLister.spyCompleted.wait(500)); 1781 QVERIFY(dirLister.isFinished()); 1782 QVERIFY(KDirWatch::self()->contains(newDirPath)); 1783 } 1784 1785 // watches were removed as the dirItem were evicted from cache 1786 QVERIFY(!KDirWatch::self()->contains(newDir.path())); 1787 QVERIFY(!KDirWatch::self()->contains(newDir.path() + QString("dir_0"))); 1788 QVERIFY(KDirWatch::self()->contains(newDir.path() + QString("dir_1"))); 1789 } 1790 1791 #include "moc_kdirlistertest.cpp"