| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250 |
- /*
- * This software is in the public domain, furnished "as is", without technical
- * support, and with no warranty, express or implied, as to its usefulness for
- * any purpose.
- *
- */
- #include <QtTest>
- #include "syncenginetestutils.h"
- #include "common/vfs.h"
- #include "config.h"
- #include <syncengine.h>
- #include "vfs/cfapi/cfapiwrapper.h"
- namespace cfapi {
- using namespace OCC::CfApiWrapper;
- }
- #define CFVERIFY_VIRTUAL(folder, path) \
- QVERIFY(QFileInfo((folder).localPath() + (path)).exists()); \
- QVERIFY(cfapi::isSparseFile((folder).localPath() + (path))); \
- QVERIFY(dbRecord((folder), (path)).isValid()); \
- QCOMPARE(dbRecord((folder), (path))._type, ItemTypeVirtualFile);
- #define CFVERIFY_NONVIRTUAL(folder, path) \
- QVERIFY(QFileInfo((folder).localPath() + (path)).exists()); \
- QVERIFY(!cfapi::isSparseFile((folder).localPath() + (path))); \
- QVERIFY(dbRecord((folder), (path)).isValid()); \
- QCOMPARE(dbRecord((folder), (path))._type, ItemTypeFile);
- #define CFVERIFY_GONE(folder, path) \
- QVERIFY(!QFileInfo((folder).localPath() + (path)).exists()); \
- QVERIFY(!dbRecord((folder), (path)).isValid());
- using namespace OCC;
- enum ErrorKind : int {
- NoError = 0,
- // Lower code are corresponding to HTTP error code
- Timeout = 1000,
- };
- void setPinState(const QString &path, PinState state, cfapi::SetPinRecurseMode mode)
- {
- Q_ASSERT(mode == cfapi::Recurse || mode == cfapi::NoRecurse);
- const auto p = QDir::toNativeSeparators(path);
- const auto handle = cfapi::handleForPath(p);
- Q_ASSERT(handle);
- const auto result = cfapi::setPinState(p, state, mode);
- Q_ASSERT(result);
- if (mode == cfapi::NoRecurse) {
- const auto result = cfapi::setPinState(p, PinState::Inherited, cfapi::ChildrenOnly);
- Q_ASSERT(result);
- }
- }
- bool itemInstruction(const ItemCompletedSpy &spy, const QString &path, const SyncInstructions instr)
- {
- auto item = spy.findItem(path);
- return item->_instruction == instr;
- }
- SyncJournalFileRecord dbRecord(FakeFolder &folder, const QString &path)
- {
- SyncJournalFileRecord record;
- [[maybe_unused]] const auto result = folder.syncJournal().getFileRecord(path, &record);
- return record;
- }
- void triggerDownload(FakeFolder &folder, const QByteArray &path)
- {
- auto &journal = folder.syncJournal();
- SyncJournalFileRecord record;
- if (!journal.getFileRecord(path, &record) || !record.isValid()) {
- return;
- }
- record._type = ItemTypeVirtualFileDownload;
- QVERIFY(journal.setFileRecord(record));
- journal.schedulePathForRemoteDiscovery(record._path);
- }
- void markForDehydration(FakeFolder &folder, const QByteArray &path)
- {
- auto &journal = folder.syncJournal();
- SyncJournalFileRecord record;
- if (!journal.getFileRecord(path, &record) || !record.isValid()) {
- return;
- }
- record._type = ItemTypeVirtualFileDehydration;
- QVERIFY(journal.setFileRecord(record));
- journal.schedulePathForRemoteDiscovery(record._path);
- }
- QSharedPointer<Vfs> setupVfs(FakeFolder &folder)
- {
- auto cfapiVfs = QSharedPointer<Vfs>(createVfsFromPlugin(Vfs::WindowsCfApi).release());
- QObject::connect(&folder.syncEngine().syncFileStatusTracker(), &SyncFileStatusTracker::fileStatusChanged,
- cfapiVfs.data(), &Vfs::fileStatusChanged);
- folder.switchToVfs(cfapiVfs);
- ::setPinState(folder.localPath(), PinState::Unspecified, cfapi::NoRecurse);
- return cfapiVfs;
- }
- class TestSyncCfApi : public QObject
- {
- Q_OBJECT
- private slots:
- void testVirtualFileLifecycle_data()
- {
- QTest::addColumn<bool>("doLocalDiscovery");
- QTest::newRow("full local discovery") << true;
- QTest::newRow("skip local discovery") << false;
- }
- void testVirtualFileLifecycle()
- {
- QFETCH(bool, doLocalDiscovery);
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- if (!doLocalDiscovery)
- fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::DatabaseAndFilesystem);
- };
- cleanup();
- // Create a virtual file for a new remote file
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().insert("A/a1", 64);
- auto someDate = QDateTime(QDate(1984, 07, 30), QTime(1,3,2));
- fakeFolder.remoteModifier().setModTime("A/a1", someDate);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 64);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").lastModified(), someDate);
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_NEW));
- cleanup();
- // Another sync doesn't actually lead to changes
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 64);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").lastModified(), someDate);
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(completeSpy.isEmpty());
- cleanup();
- // Not even when the remote is rediscovered
- fakeFolder.syncJournal().forceRemoteDiscoveryNextSync();
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 64);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").lastModified(), someDate);
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(completeSpy.isEmpty());
- cleanup();
- // Neither does a remote change
- fakeFolder.remoteModifier().appendByte("A/a1");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 65);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").lastModified(), someDate);
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_UPDATE_METADATA));
- QCOMPARE(dbRecord(fakeFolder, "A/a1")._fileSize, 65);
- cleanup();
- // If the local virtual file is removed, this will be propagated remotely
- if (!doLocalDiscovery)
- fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::DatabaseAndFilesystem, { "A" });
- fakeFolder.localModifier().remove("A/a1");
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
- QVERIFY(!fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(!dbRecord(fakeFolder, "A/a1").isValid());
- cleanup();
- // Recreate a1 before carrying on with the other tests
- fakeFolder.remoteModifier().insert("A/a1", 65);
- fakeFolder.remoteModifier().setModTime("A/a1", someDate);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 65);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").lastModified(), someDate);
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_NEW));
- cleanup();
- // Remote rename is propagated
- fakeFolder.remoteModifier().rename("A/a1", "A/a1m");
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "A/a1").exists());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1m");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1m").size(), 65);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1m").lastModified(), someDate);
- QVERIFY(!fakeFolder.currentRemoteState().find("A/a1"));
- QVERIFY(fakeFolder.currentRemoteState().find("A/a1m"));
- QVERIFY(
- itemInstruction(completeSpy, "A/a1m", CSYNC_INSTRUCTION_RENAME)
- || (itemInstruction(completeSpy, "A/a1m", CSYNC_INSTRUCTION_NEW)
- && itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_REMOVE)));
- QVERIFY(!dbRecord(fakeFolder, "A/a1").isValid());
- cleanup();
- // Remote remove is propagated
- fakeFolder.remoteModifier().remove("A/a1m");
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "A/a1m").exists());
- QVERIFY(!fakeFolder.currentRemoteState().find("A/a1m"));
- QVERIFY(itemInstruction(completeSpy, "A/a1m", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(!dbRecord(fakeFolder, "A/a1").isValid());
- QVERIFY(!dbRecord(fakeFolder, "A/a1m").isValid());
- cleanup();
- // Edge case: Local virtual file but no db entry for some reason
- fakeFolder.remoteModifier().insert("A/a2", 32);
- fakeFolder.remoteModifier().insert("A/a3", 33);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a2").size(), 32);
- CFVERIFY_VIRTUAL(fakeFolder, "A/a3");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a3").size(), 33);
- cleanup();
- QVERIFY(fakeFolder.syncEngine().journal()->deleteFileRecord("A/a2"));
- QVERIFY(fakeFolder.syncEngine().journal()->deleteFileRecord("A/a3"));
- fakeFolder.remoteModifier().remove("A/a3");
- fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::FilesystemOnly);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a2").size(), 32);
- QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_UPDATE_METADATA));
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "A/a3").exists());
- QVERIFY(itemInstruction(completeSpy, "A/a3", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(!dbRecord(fakeFolder, "A/a3").isValid());
- cleanup();
- }
- void testVirtualFileConflict()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- // Create a virtual file for a new remote file
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().insert("A/a1", 11);
- fakeFolder.remoteModifier().insert("A/a2", 12);
- fakeFolder.remoteModifier().mkdir("B");
- fakeFolder.remoteModifier().insert("B/b1", 21);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_VIRTUAL(fakeFolder, "B/b1");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a1").size(), 11);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/a2").size(), 12);
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "B/b1").size(), 21);
- cleanup();
- // All the files are touched on the server
- fakeFolder.remoteModifier().appendByte("A/a1");
- fakeFolder.remoteModifier().appendByte("A/a2");
- fakeFolder.remoteModifier().appendByte("B/b1");
- // A: the correct file and a conflicting file are added
- // B: user adds a *directory* locally
- fakeFolder.localModifier().remove("A/a1");
- fakeFolder.localModifier().insert("A/a1", 12);
- fakeFolder.localModifier().remove("A/a2");
- fakeFolder.localModifier().insert("A/a2", 10);
- fakeFolder.localModifier().remove("B/b1");
- fakeFolder.localModifier().mkdir("B/b1");
- fakeFolder.localModifier().insert("B/b1/foo");
- QVERIFY(fakeFolder.syncOnce());
- // Everything is CONFLICT
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_CONFLICT));
- QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_CONFLICT));
- QVERIFY(itemInstruction(completeSpy, "B/b1", CSYNC_INSTRUCTION_CONFLICT));
- // conflict files should exist
- QCOMPARE(fakeFolder.syncJournal().conflictRecordPaths().size(), 2);
- // nothing should have the virtual file tag
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_NONVIRTUAL(fakeFolder, "B/b1");
- cleanup();
- }
- void testWithNormalSync()
- {
- FakeFolder fakeFolder{ FileInfo::A12_B12_C12_S12() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- // No effect sync
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- cleanup();
- // Existing files are propagated just fine in both directions
- fakeFolder.localModifier().appendByte("A/a1");
- fakeFolder.localModifier().insert("A/a3");
- fakeFolder.remoteModifier().appendByte("A/a2");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- cleanup();
- // New files on the remote create virtual files
- fakeFolder.remoteModifier().insert("A/new", 42);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/new");
- QCOMPARE(QFileInfo(fakeFolder.localPath() + "A/new").size(), 42);
- QVERIFY(fakeFolder.currentRemoteState().find("A/new"));
- QVERIFY(itemInstruction(completeSpy, "A/new", CSYNC_INSTRUCTION_NEW));
- cleanup();
- }
- void testVirtualFileDownload()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- // Create a virtual file for remote files
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().insert("A/a1");
- fakeFolder.remoteModifier().insert("A/a2");
- fakeFolder.remoteModifier().insert("A/a3");
- fakeFolder.remoteModifier().insert("A/a4");
- fakeFolder.remoteModifier().insert("A/a5");
- fakeFolder.remoteModifier().insert("A/a6");
- fakeFolder.remoteModifier().insert("A/a7");
- fakeFolder.remoteModifier().insert("A/b1");
- fakeFolder.remoteModifier().insert("A/b2");
- fakeFolder.remoteModifier().insert("A/b3");
- fakeFolder.remoteModifier().insert("A/b4");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a3");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a4");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a5");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a6");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a7");
- CFVERIFY_VIRTUAL(fakeFolder, "A/b1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/b2");
- CFVERIFY_VIRTUAL(fakeFolder, "A/b3");
- CFVERIFY_VIRTUAL(fakeFolder, "A/b4");
- cleanup();
- // Download by changing the db entry
- triggerDownload(fakeFolder, "A/a1");
- triggerDownload(fakeFolder, "A/a2");
- triggerDownload(fakeFolder, "A/a3");
- triggerDownload(fakeFolder, "A/a4");
- triggerDownload(fakeFolder, "A/a5");
- triggerDownload(fakeFolder, "A/a6");
- triggerDownload(fakeFolder, "A/a7");
- triggerDownload(fakeFolder, "A/b1");
- triggerDownload(fakeFolder, "A/b2");
- triggerDownload(fakeFolder, "A/b3");
- triggerDownload(fakeFolder, "A/b4");
- // Remote complications
- fakeFolder.remoteModifier().appendByte("A/a2");
- fakeFolder.remoteModifier().remove("A/a3");
- fakeFolder.remoteModifier().rename("A/a4", "A/a4m");
- fakeFolder.remoteModifier().appendByte("A/b2");
- fakeFolder.remoteModifier().remove("A/b3");
- fakeFolder.remoteModifier().rename("A/b4", "A/b4m");
- // Local complications
- fakeFolder.localModifier().remove("A/a5");
- fakeFolder.localModifier().insert("A/a5");
- fakeFolder.localModifier().remove("A/a6");
- fakeFolder.localModifier().insert("A/a6");
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_SYNC));
- QCOMPARE(completeSpy.findItem("A/a1")->_type, ItemTypeVirtualFileDownload);
- QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_SYNC));
- QCOMPARE(completeSpy.findItem("A/a2")->_type, ItemTypeVirtualFileDownload);
- QVERIFY(itemInstruction(completeSpy, "A/a3", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(itemInstruction(completeSpy, "A/a4m", CSYNC_INSTRUCTION_NEW));
- QVERIFY(itemInstruction(completeSpy, "A/a4", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(itemInstruction(completeSpy, "A/a5", CSYNC_INSTRUCTION_CONFLICT));
- QVERIFY(itemInstruction(completeSpy, "A/a6", CSYNC_INSTRUCTION_CONFLICT));
- QVERIFY(itemInstruction(completeSpy, "A/a7", CSYNC_INSTRUCTION_SYNC));
- QVERIFY(itemInstruction(completeSpy, "A/b1", CSYNC_INSTRUCTION_SYNC));
- QVERIFY(itemInstruction(completeSpy, "A/b2", CSYNC_INSTRUCTION_SYNC));
- QVERIFY(itemInstruction(completeSpy, "A/b3", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(itemInstruction(completeSpy, "A/b4m", CSYNC_INSTRUCTION_NEW));
- QVERIFY(itemInstruction(completeSpy, "A/b4", CSYNC_INSTRUCTION_REMOVE));
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_GONE(fakeFolder, "A/a3");
- CFVERIFY_GONE(fakeFolder, "A/a4");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a4m");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a5");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a6");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a7");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/b1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/b2");
- CFVERIFY_GONE(fakeFolder, "A/b3");
- CFVERIFY_GONE(fakeFolder, "A/b4");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/b4m");
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- }
- void testVirtualFileDownloadResume()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- QVERIFY(fakeFolder.syncJournal().wipeErrorBlacklist() != -1);
- };
- cleanup();
- // Create a virtual file for remote files
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().insert("A/a1");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- cleanup();
- // Download by changing the db entry
- triggerDownload(fakeFolder, "A/a1");
- fakeFolder.serverErrorPaths().append("A/a1", 500);
- QVERIFY(!fakeFolder.syncOnce());
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_SYNC));
- QVERIFY(cfapi::isSparseFile(fakeFolder.localPath() + "A/a1"));
- QVERIFY(QFileInfo(fakeFolder.localPath() + "A/a1").exists());
- QCOMPARE(dbRecord(fakeFolder, "A/a1")._type, ItemTypeVirtualFileDownload);
- cleanup();
- fakeFolder.serverErrorPaths().clear();
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_SYNC));
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a1");
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- }
- void testNewFilesNotVirtual()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().insert("A/a1");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- ::setPinState(fakeFolder.localPath(), PinState::AlwaysLocal, cfapi::NoRecurse);
- // Create a new remote file, it'll not be virtual
- fakeFolder.remoteModifier().insert("A/a2");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a2");
- }
- void testDownloadRecursive()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- // Create a virtual file for remote files
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().mkdir("A/Sub");
- fakeFolder.remoteModifier().mkdir("A/Sub/SubSub");
- fakeFolder.remoteModifier().mkdir("A/Sub2");
- fakeFolder.remoteModifier().mkdir("B");
- fakeFolder.remoteModifier().mkdir("B/Sub");
- fakeFolder.remoteModifier().insert("A/a1");
- fakeFolder.remoteModifier().insert("A/a2");
- fakeFolder.remoteModifier().insert("A/Sub/a3");
- fakeFolder.remoteModifier().insert("A/Sub/a4");
- fakeFolder.remoteModifier().insert("A/Sub/SubSub/a5");
- fakeFolder.remoteModifier().insert("A/Sub2/a6");
- fakeFolder.remoteModifier().insert("B/b1");
- fakeFolder.remoteModifier().insert("B/Sub/b2");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub/a3");
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub/a4");
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub/SubSub/a5");
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub2/a6");
- CFVERIFY_VIRTUAL(fakeFolder, "B/b1");
- CFVERIFY_VIRTUAL(fakeFolder, "B/Sub/b2");
- // Download All file in the directory A/Sub
- // (as in Folder::downloadVirtualFile)
- fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("A/Sub");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/a3");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/a4");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/SubSub/a5");
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub2/a6");
- CFVERIFY_VIRTUAL(fakeFolder, "B/b1");
- CFVERIFY_VIRTUAL(fakeFolder, "B/Sub/b2");
- // Add a file in a subfolder that was downloaded
- // Currently, this continue to add it as a virtual file.
- fakeFolder.remoteModifier().insert("A/Sub/SubSub/a7");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "A/Sub/SubSub/a7");
- // Now download all files in "A"
- fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("A");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/a2");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/a3");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/a4");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/SubSub/a5");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub/SubSub/a7");
- CFVERIFY_NONVIRTUAL(fakeFolder, "A/Sub2/a6");
- CFVERIFY_VIRTUAL(fakeFolder, "B/b1");
- CFVERIFY_VIRTUAL(fakeFolder, "B/Sub/b2");
- // Now download remaining files in "B"
- fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("B");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- }
- void testRenameVirtual()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- fakeFolder.remoteModifier().insert("file1", 128, 'C');
- fakeFolder.remoteModifier().insert("file2", 256, 'C');
- fakeFolder.remoteModifier().insert("file3", 256, 'C');
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "file1");
- CFVERIFY_VIRTUAL(fakeFolder, "file2");
- CFVERIFY_VIRTUAL(fakeFolder, "file3");
- cleanup();
- fakeFolder.localModifier().rename("file1", "renamed1");
- fakeFolder.localModifier().rename("file2", "renamed2");
- triggerDownload(fakeFolder, "file2");
- triggerDownload(fakeFolder, "file3");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_GONE(fakeFolder, "file1");
- CFVERIFY_VIRTUAL(fakeFolder, "renamed1");
- QVERIFY(fakeFolder.currentRemoteState().find("renamed1"));
- QVERIFY(itemInstruction(completeSpy, "renamed1", CSYNC_INSTRUCTION_RENAME));
- // file2 has a conflict between the download request and the rename:
- // the rename wins, the download is ignored
- CFVERIFY_GONE(fakeFolder, "file2");
- CFVERIFY_VIRTUAL(fakeFolder, "renamed2");
- QVERIFY(fakeFolder.currentRemoteState().find("renamed2"));
- QVERIFY(itemInstruction(completeSpy, "renamed2", CSYNC_INSTRUCTION_RENAME));
- QVERIFY(itemInstruction(completeSpy, "file3", CSYNC_INSTRUCTION_SYNC));
- CFVERIFY_NONVIRTUAL(fakeFolder, "file3");
- cleanup();
- }
- void testRenameVirtual2()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- fakeFolder.remoteModifier().insert("case3", 128, 'C');
- fakeFolder.remoteModifier().insert("case4", 256, 'C');
- QVERIFY(fakeFolder.syncOnce());
- triggerDownload(fakeFolder, "case4");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "case3");
- CFVERIFY_NONVIRTUAL(fakeFolder, "case4");
- cleanup();
- // Case 1: non-virtual, foo -> bar (tested elsewhere)
- // Case 2: virtual, foo -> bar (tested elsewhere)
- // Case 3: virtual, foo.oc -> bar.oc (db hydrate)
- fakeFolder.localModifier().rename("case3", "case3-rename");
- triggerDownload(fakeFolder, "case3");
- // Case 4: non-virtual foo -> bar (db dehydrate)
- fakeFolder.localModifier().rename("case4", "case4-rename");
- markForDehydration(fakeFolder, "case4");
- QVERIFY(fakeFolder.syncOnce());
- // Case 3: the rename went though, hydration is forgotten
- CFVERIFY_GONE(fakeFolder, "case3");
- CFVERIFY_VIRTUAL(fakeFolder, "case3-rename");
- QVERIFY(!fakeFolder.currentRemoteState().find("case3"));
- QVERIFY(fakeFolder.currentRemoteState().find("case3-rename"));
- QVERIFY(itemInstruction(completeSpy, "case3-rename", CSYNC_INSTRUCTION_RENAME));
- // Case 4: the rename went though, dehydration is forgotten
- CFVERIFY_GONE(fakeFolder, "case4");
- CFVERIFY_NONVIRTUAL(fakeFolder, "case4-rename");
- QVERIFY(!fakeFolder.currentRemoteState().find("case4"));
- QVERIFY(fakeFolder.currentRemoteState().find("case4-rename"));
- QVERIFY(itemInstruction(completeSpy, "case4-rename", CSYNC_INSTRUCTION_RENAME));
- }
- // Dehydration via sync works
- void testSyncDehydration()
- {
- FakeFolder fakeFolder{ FileInfo::A12_B12_C12_S12() };
- // empty files would not work, so, we're gonna remove and re-insert them with 1MB data
- fakeFolder.remoteModifier().remove("A/a1");
- fakeFolder.remoteModifier().remove("A/a2");
- fakeFolder.remoteModifier().insert("A/a1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("A/a2", 1024 * 1024);
- setupVfs(fakeFolder);
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- //
- // Mark for dehydration and check
- //
- QVERIFY(fakeFolder.currentLocalState().find("A/a1"));
- markForDehydration(fakeFolder, "A/a1");
- markForDehydration(fakeFolder, "A/a2");
- fakeFolder.remoteModifier().appendByte("A/a2");
- // expect: normal dehydration
- markForDehydration(fakeFolder, "B/b1");
- fakeFolder.remoteModifier().remove("B/b1");
- // expect: local removal
- markForDehydration(fakeFolder, "B/b2");
- fakeFolder.remoteModifier().rename("B/b2", "B/b3");
- // expect: B/b2 is gone, B/b3 is NEW placeholder
- markForDehydration(fakeFolder, "C/c1");
- fakeFolder.localModifier().appendByte("C/c1");
- // expect: no dehydration, upload of c1
- markForDehydration(fakeFolder, "C/c2");
- fakeFolder.localModifier().appendByte("C/c2");
- fakeFolder.remoteModifier().appendByte("C/c2");
- fakeFolder.remoteModifier().appendByte("C/c2");
- // expect: no dehydration, conflict
- QVERIFY(fakeFolder.syncOnce());
- auto isDehydrated = [&](const QString &path) {
- return cfapi::isSparseFile(fakeFolder.localPath() + path)
- && QFileInfo(fakeFolder.localPath() + path).exists();
- };
- auto hasDehydratedDbEntries = [&](const QString &path) {
- SyncJournalFileRecord rec;
- return fakeFolder.syncJournal().getFileRecord(path, &rec) && rec.isValid() && rec._type == ItemTypeVirtualFile;
- };
- QVERIFY(isDehydrated("A/a1"));
- QVERIFY(hasDehydratedDbEntries("A/a1"));
- QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_SYNC));
- QCOMPARE(completeSpy.findItem("A/a1")->_type, ItemTypeVirtualFileDehydration);
- QCOMPARE(completeSpy.findItem("A/a1")->_file, QStringLiteral("A/a1"));
- QVERIFY(isDehydrated("A/a2"));
- QVERIFY(hasDehydratedDbEntries("A/a2"));
- QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_SYNC));
- QCOMPARE(completeSpy.findItem("A/a2")->_type, ItemTypeVirtualFileDehydration);
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "B/b1").exists());
- QVERIFY(!fakeFolder.currentRemoteState().find("B/b1"));
- QVERIFY(itemInstruction(completeSpy, "B/b1", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "B/b2").exists());
- QVERIFY(!fakeFolder.currentRemoteState().find("B/b2"));
- QVERIFY(isDehydrated("B/b3"));
- QVERIFY(hasDehydratedDbEntries("B/b3"));
- QVERIFY(itemInstruction(completeSpy, "B/b2", CSYNC_INSTRUCTION_REMOVE));
- QVERIFY(itemInstruction(completeSpy, "B/b3", CSYNC_INSTRUCTION_NEW));
- QCOMPARE(fakeFolder.currentRemoteState().find("C/c1")->size, 25);
- QVERIFY(itemInstruction(completeSpy, "C/c1", CSYNC_INSTRUCTION_SYNC));
- QCOMPARE(fakeFolder.currentRemoteState().find("C/c2")->size, 26);
- QVERIFY(itemInstruction(completeSpy, "C/c2", CSYNC_INSTRUCTION_CONFLICT));
- cleanup();
- auto expectedRemoteState = fakeFolder.currentRemoteState();
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentRemoteState(), expectedRemoteState);
- QVERIFY(isDehydrated("A/a1"));
- QVERIFY(hasDehydratedDbEntries("A/a1"));
- QVERIFY(isDehydrated("A/a2"));
- QVERIFY(hasDehydratedDbEntries("A/a2"));
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "B/b1").exists());
- QVERIFY(!QFileInfo(fakeFolder.localPath() + "B/b2").exists());
- QVERIFY(isDehydrated("B/b3"));
- QVERIFY(hasDehydratedDbEntries("B/b3"));
- QVERIFY(QFileInfo(fakeFolder.localPath() + "C/c1").exists());
- QVERIFY(dbRecord(fakeFolder, "C/c1").isValid());
- QVERIFY(!isDehydrated("C/c1"));
- QVERIFY(!hasDehydratedDbEntries("C/c1"));
- QVERIFY(QFileInfo(fakeFolder.localPath() + "C/c2").exists());
- QVERIFY(dbRecord(fakeFolder, "C/c2").isValid());
- QVERIFY(!isDehydrated("C/c2"));
- QVERIFY(!hasDehydratedDbEntries("C/c2"));
- }
- void testWipeVirtualSuffixFiles()
- {
- // TODO: Part of this test related to A/a3 is always failing on CI but never fails locally
- // I had to comment it out as this prevents from running all other tests with no working ways to fix that
- FakeFolder fakeFolder{ FileInfo{} };
- setupVfs(fakeFolder);
- // Create a suffix-vfs baseline
- fakeFolder.remoteModifier().mkdir("A");
- fakeFolder.remoteModifier().mkdir("A/B");
- fakeFolder.remoteModifier().insert("f1");
- fakeFolder.remoteModifier().insert("A/a1");
- // fakeFolder.remoteModifier().insert("A/a3");
- fakeFolder.remoteModifier().insert("A/B/b1");
- fakeFolder.localModifier().mkdir("A");
- fakeFolder.localModifier().mkdir("A/B");
- fakeFolder.localModifier().insert("f2");
- fakeFolder.localModifier().insert("A/a2");
- fakeFolder.localModifier().insert("A/B/b2");
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "f1");
- CFVERIFY_VIRTUAL(fakeFolder, "A/a1");
- // CFVERIFY_VIRTUAL(fakeFolder, "A/a3");
- CFVERIFY_VIRTUAL(fakeFolder, "A/B/b1");
- // Make local changes to a3
- // fakeFolder.localModifier().remove("A/a3");
- // fakeFolder.localModifier().insert("A/a3", 100);
- // Now wipe the virtuals
- SyncEngine::wipeVirtualFiles(fakeFolder.localPath(), fakeFolder.syncJournal(), *fakeFolder.syncEngine().syncOptions()._vfs);
- CFVERIFY_GONE(fakeFolder, "f1");
- CFVERIFY_GONE(fakeFolder, "A/a1");
- //QVERIFY(QFileInfo(fakeFolder.localPath() + "A/a3").exists());
- // QVERIFY(!dbRecord(fakeFolder, "A/a3").isValid());
- CFVERIFY_GONE(fakeFolder, "A/B/b1");
- fakeFolder.switchToVfs(QSharedPointer<Vfs>(new VfsOff));
- // ItemCompletedSpy completeSpy(fakeFolder);
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(fakeFolder.currentLocalState().find("A"));
- QVERIFY(fakeFolder.currentLocalState().find("A/B"));
- QVERIFY(fakeFolder.currentLocalState().find("A/B/b1"));
- QVERIFY(fakeFolder.currentLocalState().find("A/B/b2"));
- QVERIFY(fakeFolder.currentLocalState().find("A/a1"));
- QVERIFY(fakeFolder.currentLocalState().find("A/a2"));
- // QVERIFY(fakeFolder.currentLocalState().find("A/a3"));
- QVERIFY(fakeFolder.currentLocalState().find("f1"));
- QVERIFY(fakeFolder.currentLocalState().find("f2"));
- // a3 has a conflict
- // QVERIFY(itemInstruction(completeSpy, "A/a3", CSYNC_INSTRUCTION_CONFLICT));
- // conflict files should exist
- // QCOMPARE(fakeFolder.syncJournal().conflictRecordPaths().size(), 1);
- }
- void testNewVirtuals()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("local");
- fakeFolder.remoteModifier().mkdir("online");
- fakeFolder.remoteModifier().mkdir("unspec");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "unspec", PinState::Unspecified, cfapi::Recurse);
- // Test 1: root is Unspecified
- fakeFolder.remoteModifier().insert("file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("online/file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("local/file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("unspec/file1", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "file1");
- CFVERIFY_VIRTUAL(fakeFolder, "online/file1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "local/file1");
- CFVERIFY_VIRTUAL(fakeFolder, "unspec/file1");
- // Test 2: change root to AlwaysLocal
- ::setPinState(fakeFolder.localPath(), PinState::AlwaysLocal, cfapi::Recurse);
- // Need to force pin state for the subfolders again
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "unspec", PinState::Unspecified, cfapi::Recurse);
- fakeFolder.remoteModifier().insert("file2", 1024 * 1024);
- fakeFolder.remoteModifier().insert("online/file2", 1024 * 1024);
- fakeFolder.remoteModifier().insert("local/file2", 1024 * 1024);
- fakeFolder.remoteModifier().insert("unspec/file2", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_NONVIRTUAL(fakeFolder, "file2");
- CFVERIFY_VIRTUAL(fakeFolder, "online/file2");
- CFVERIFY_NONVIRTUAL(fakeFolder, "local/file2");
- CFVERIFY_VIRTUAL(fakeFolder, "unspec/file2");
- // root file1 was hydrated due to its new pin state
- CFVERIFY_NONVIRTUAL(fakeFolder, "file1");
- // file1 is unchanged in the explicitly pinned subfolders
- CFVERIFY_VIRTUAL(fakeFolder, "online/file1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "local/file1");
- CFVERIFY_VIRTUAL(fakeFolder, "unspec/file1");
- // Test 3: change root to OnlineOnly
- ::setPinState(fakeFolder.localPath(), PinState::OnlineOnly, cfapi::Recurse);
- // Need to force pin state for the subfolders again
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "unspec", PinState::Unspecified, cfapi::Recurse);
- fakeFolder.remoteModifier().insert("file3", 1024 * 1024);
- fakeFolder.remoteModifier().insert("online/file3", 1024 * 1024);
- fakeFolder.remoteModifier().insert("local/file3", 1024 * 1024);
- fakeFolder.remoteModifier().insert("unspec/file3", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "file3");
- CFVERIFY_VIRTUAL(fakeFolder, "online/file3");
- CFVERIFY_NONVIRTUAL(fakeFolder, "local/file3");
- CFVERIFY_VIRTUAL(fakeFolder, "unspec/file3");
- // root file1 was dehydrated due to its new pin state
- CFVERIFY_VIRTUAL(fakeFolder, "file1");
- // file1 is unchanged in the explicitly pinned subfolders
- CFVERIFY_VIRTUAL(fakeFolder, "online/file1");
- CFVERIFY_NONVIRTUAL(fakeFolder, "local/file1");
- CFVERIFY_VIRTUAL(fakeFolder, "unspec/file1");
- }
- void testAvailability()
- {
- FakeFolder fakeFolder{ FileInfo() };
- auto vfs = setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("local");
- fakeFolder.remoteModifier().mkdir("local/sub");
- fakeFolder.remoteModifier().mkdir("online");
- fakeFolder.remoteModifier().mkdir("online/sub");
- fakeFolder.remoteModifier().mkdir("unspec");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::Recurse);
- ::setPinState(fakeFolder.localPath() + "unspec", PinState::Unspecified, cfapi::Recurse);
- fakeFolder.remoteModifier().insert("file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("online/file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("online/file2", 1024 * 1024);
- fakeFolder.remoteModifier().insert("local/file1", 1024 * 1024);
- fakeFolder.remoteModifier().insert("local/file2", 1024 * 1024);
- fakeFolder.remoteModifier().insert("unspec/file1", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- // root is unspecified
- QCOMPARE(*vfs->availability("file1"), VfsItemAvailability::AllDehydrated);
- QCOMPARE(*vfs->availability("local"), VfsItemAvailability::AlwaysLocal);
- QCOMPARE(*vfs->availability("local/file1"), VfsItemAvailability::AlwaysLocal);
- QCOMPARE(*vfs->availability("online"), VfsItemAvailability::OnlineOnly);
- QCOMPARE(*vfs->availability("online/file1"), VfsItemAvailability::OnlineOnly);
- QCOMPARE(*vfs->availability("unspec"), VfsItemAvailability::AllDehydrated);
- QCOMPARE(*vfs->availability("unspec/file1"), VfsItemAvailability::AllDehydrated);
- // Subitem pin states can ruin "pure" availabilities
- ::setPinState(fakeFolder.localPath() + "local/sub", PinState::OnlineOnly, cfapi::NoRecurse);
- QCOMPARE(*vfs->availability("local"), VfsItemAvailability::AllHydrated);
- ::setPinState(fakeFolder.localPath() + "online/sub", PinState::Unspecified, cfapi::NoRecurse);
- QCOMPARE(*vfs->availability("online"), VfsItemAvailability::AllDehydrated);
- triggerDownload(fakeFolder, "unspec/file1");
- ::setPinState(fakeFolder.localPath() + "local/file2", PinState::OnlineOnly, cfapi::NoRecurse);
- ::setPinState(fakeFolder.localPath() + "online/file2", PinState::AlwaysLocal, cfapi::NoRecurse);
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->availability("unspec"), VfsItemAvailability::AllHydrated);
- QCOMPARE(*vfs->availability("local"), VfsItemAvailability::Mixed);
- QCOMPARE(*vfs->availability("online"), VfsItemAvailability::Mixed);
- QVERIFY(vfs->setPinState("local", PinState::AlwaysLocal));
- QVERIFY(vfs->setPinState("online", PinState::OnlineOnly));
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->availability("online"), VfsItemAvailability::OnlineOnly);
- QCOMPARE(*vfs->availability("local"), VfsItemAvailability::AlwaysLocal);
- auto r = vfs->availability("nonexistent");
- QVERIFY(!r);
- QCOMPARE(r.error(), Vfs::AvailabilityError::NoSuchItem);
- }
- void testPinStateLocals()
- {
- FakeFolder fakeFolder{ FileInfo() };
- auto vfs = setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("local");
- fakeFolder.remoteModifier().mkdir("online");
- fakeFolder.remoteModifier().mkdir("unspec");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::NoRecurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::NoRecurse);
- ::setPinState(fakeFolder.localPath() + "unspec", PinState::Unspecified, cfapi::NoRecurse);
- fakeFolder.localModifier().insert("file1", 1024 * 1024);
- fakeFolder.localModifier().insert("online/file1", 1024 * 1024);
- fakeFolder.localModifier().insert("online/file2", 1024 * 1024);
- fakeFolder.localModifier().insert("local/file1", 1024 * 1024);
- fakeFolder.localModifier().insert("unspec/file1", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- // root is unspecified
- QCOMPARE(*vfs->pinState("file1"), PinState::Unspecified);
- QCOMPARE(*vfs->pinState("local/file1"), PinState::AlwaysLocal);
- QCOMPARE(*vfs->pinState("online/file1"), PinState::Unspecified);
- QCOMPARE(*vfs->pinState("unspec/file1"), PinState::Unspecified);
- // Sync again: bad pin states of new local files usually take effect on second sync
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- // When a file in an online-only folder is renamed, it retains its pin
- fakeFolder.localModifier().rename("online/file1", "online/file1rename");
- fakeFolder.remoteModifier().rename("online/file2", "online/file2rename");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->pinState("online/file1rename"), PinState::Unspecified);
- QCOMPARE(*vfs->pinState("online/file2rename"), PinState::Unspecified);
- // When a folder is renamed, the pin states inside should be retained
- fakeFolder.localModifier().rename("online", "onlinerenamed1");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->pinState("onlinerenamed1"), PinState::OnlineOnly);
- QCOMPARE(*vfs->pinState("onlinerenamed1/file1rename"), PinState::Unspecified);
- fakeFolder.remoteModifier().rename("onlinerenamed1", "onlinerenamed2");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->pinState("onlinerenamed2"), PinState::OnlineOnly);
- QCOMPARE(*vfs->pinState("onlinerenamed2/file1rename"), PinState::Unspecified);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- // When a file is deleted and later a new file has the same name, the old pin
- // state isn't preserved.
- QCOMPARE(*vfs->pinState("onlinerenamed2/file1rename"), PinState::Unspecified);
- fakeFolder.remoteModifier().remove("onlinerenamed2/file1rename");
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(!vfs->pinState("onlinerenamed2/file1rename"));
- fakeFolder.remoteModifier().insert("onlinerenamed2/file1rename", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(*vfs->pinState("onlinerenamed2/file1rename"), PinState::OnlineOnly);
- // When a file is hydrated or dehydrated due to pin state it retains its pin state
- QVERIFY(vfs->setPinState("onlinerenamed2/file1rename", PinState::AlwaysLocal));
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(fakeFolder.currentLocalState().find("onlinerenamed2/file1rename"));
- QCOMPARE(*vfs->pinState("onlinerenamed2/file1rename"), PinState::AlwaysLocal);
- QVERIFY(vfs->setPinState("onlinerenamed2", PinState::Unspecified));
- QVERIFY(vfs->setPinState("onlinerenamed2/file1rename", PinState::OnlineOnly));
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "onlinerenamed2/file1rename");
- QCOMPARE(*vfs->pinState("onlinerenamed2/file1rename"), PinState::OnlineOnly);
- }
- void testEmptyFolderInOnlineOnlyRoot()
- {
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ItemCompletedSpy completeSpy(fakeFolder);
- auto cleanup = [&]() {
- completeSpy.clear();
- };
- cleanup();
- // OnlineOnly forced on the root
- ::setPinState(fakeFolder.localPath(), PinState::OnlineOnly, cfapi::NoRecurse);
- // No effect sync
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- cleanup();
- // Add an empty folder which should propagate
- fakeFolder.localModifier().mkdir("A");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- cleanup();
- }
- void testIncompatiblePins()
- {
- FakeFolder fakeFolder{ FileInfo() };
- auto vfs = setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("local");
- fakeFolder.remoteModifier().mkdir("online");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ::setPinState(fakeFolder.localPath() + "local", PinState::AlwaysLocal, cfapi::NoRecurse);
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::NoRecurse);
- fakeFolder.localModifier().insert("local/file1", 1024 * 1024);
- fakeFolder.localModifier().insert("online/file1", 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- ::setPinState(fakeFolder.localPath() + "local/file1", PinState::Unspecified, cfapi::Recurse);
- markForDehydration(fakeFolder, "local/file1");
- triggerDownload(fakeFolder, "online/file1");
- // the sync sets the changed files pin states to unspecified
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_NONVIRTUAL(fakeFolder, "online/file1");
- CFVERIFY_VIRTUAL(fakeFolder, "local/file1");
- QCOMPARE(*vfs->pinState("online/file1"), PinState::Unspecified);
- QCOMPARE(*vfs->pinState("local/file1"), PinState::Unspecified);
- // no change on another sync
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_NONVIRTUAL(fakeFolder, "online/file1");
- CFVERIFY_VIRTUAL(fakeFolder, "local/file1");
- }
- void testOpeningOnlineFileTriggersDownload_data()
- {
- QTest::addColumn<int>("errorKind");
- QTest::newRow("no error") << static_cast<int>(NoError);
- QTest::newRow("400") << 400;
- QTest::newRow("401") << 401;
- QTest::newRow("403") << 403;
- QTest::newRow("404") << 404;
- QTest::newRow("500") << 500;
- QTest::newRow("503") << 503;
- QTest::newRow("Timeout") << static_cast<int>(Timeout);
- }
- void testOpeningOnlineFileTriggersDownload()
- {
- QFETCH(int, errorKind);
- FakeFolder fakeFolder{ FileInfo() };
- setupVfs(fakeFolder);
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- fakeFolder.remoteModifier().mkdir("online");
- fakeFolder.remoteModifier().mkdir("online/sub");
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- ::setPinState(fakeFolder.localPath() + "online", PinState::OnlineOnly, cfapi::Recurse);
- fakeFolder.remoteModifier().insert("online/sub/file1", 10 * 1024 * 1024);
- QVERIFY(fakeFolder.syncOnce());
- CFVERIFY_VIRTUAL(fakeFolder, "online/sub/file1");
- // Setup error case if needed
- if (errorKind == Timeout) {
- fakeFolder.setServerOverride([&](QNetworkAccessManager::Operation op, const QNetworkRequest &req, QIODevice *) -> QNetworkReply * {
- if (req.url().path().endsWith("online/sub/file1")) {
- return new FakeHangingReply(op, req, this);
- }
- return nullptr;
- });
- } else if (errorKind != NoError) {
- fakeFolder.serverErrorPaths().append("online/sub/file1", errorKind);
- }
- // So the test that test timeout finishes fast
- QScopedValueRollback<int> setHttpTimeout(AbstractNetworkJob::httpTimeout, errorKind == Timeout ? 1 : 10000);
- // Simulate another process requesting the open
- QEventLoop loop;
- std::thread t([&] {
- QFile file(fakeFolder.localPath() + "online/sub/file1");
- if (file.open(QFile::ReadOnly)) {
- file.readAll();
- file.close();
- }
- QMetaObject::invokeMethod(&loop, &QEventLoop::quit, Qt::QueuedConnection);
- });
- loop.exec();
- t.detach();
- if (errorKind == NoError) {
- CFVERIFY_NONVIRTUAL(fakeFolder, "online/sub/file1");
- } else {
- CFVERIFY_VIRTUAL(fakeFolder, "online/sub/file1");
- }
- // Nothing should change
- ItemCompletedSpy completeSpy(fakeFolder);
- QVERIFY(fakeFolder.syncOnce());
- QVERIFY(completeSpy.isEmpty());
- if (errorKind == NoError) {
- CFVERIFY_NONVIRTUAL(fakeFolder, "online/sub/file1");
- } else {
- CFVERIFY_VIRTUAL(fakeFolder, "online/sub/file1");
- }
- }
- void testDataFingerPrint()
- {
- FakeFolder fakeFolder{ FileInfo{} };
- setupVfs(fakeFolder);
- fakeFolder.remoteModifier().mkdir("a");
- fakeFolder.remoteModifier().mkdir("a/b");
- fakeFolder.remoteModifier().mkdir("a/b/d");
- fakeFolder.remoteModifier().insert("a/b/otherFile.txt");
- //Server support finger print, but none is set.
- fakeFolder.remoteModifier().extraDavProperties = "<oc:data-fingerprint></oc:data-fingerprint>";
- fakeFolder.syncEngine().setLocalDiscoveryOptions(OCC::LocalDiscoveryStyle::DatabaseAndFilesystem);
- QVERIFY(fakeFolder.syncOnce());
- fakeFolder.remoteModifier().remove("a/b/otherFile.txt");
- fakeFolder.remoteModifier().remove("a/b/d");
- fakeFolder.remoteModifier().extraDavProperties = "<oc:data-fingerprint>initial_finger_print</oc:data-fingerprint>";
- fakeFolder.syncEngine().setLocalDiscoveryOptions(OCC::LocalDiscoveryStyle::DatabaseAndFilesystem);
- QVERIFY(fakeFolder.syncOnce());
- QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
- }
- };
- QTEST_GUILESS_MAIN(TestSyncCfApi)
- #include "testsynccfapi.moc"
|