testsyncvirtualfiles.cpp 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590
  1. /*
  2. * This software is in the public domain, furnished "as is", without technical
  3. * support, and with no warranty, express or implied, as to its usefulness for
  4. * any purpose.
  5. *
  6. */
  7. #include <QtTest>
  8. #include "syncenginetestutils.h"
  9. #include <syncengine.h>
  10. using namespace OCC;
  11. SyncFileItemPtr findItem(const QSignalSpy &spy, const QString &path)
  12. {
  13. for (const QList<QVariant> &args : spy) {
  14. auto item = args[0].value<SyncFileItemPtr>();
  15. if (item->destination() == path)
  16. return item;
  17. }
  18. return SyncFileItemPtr(new SyncFileItem);
  19. }
  20. bool itemInstruction(const QSignalSpy &spy, const QString &path, const csync_instructions_e instr)
  21. {
  22. auto item = findItem(spy, path);
  23. return item->_instruction == instr;
  24. }
  25. SyncJournalFileRecord dbRecord(FakeFolder &folder, const QString &path)
  26. {
  27. SyncJournalFileRecord record;
  28. folder.syncJournal().getFileRecord(path, &record);
  29. return record;
  30. }
  31. class TestSyncVirtualFiles : public QObject
  32. {
  33. Q_OBJECT
  34. private slots:
  35. void testVirtualFileLifecycle_data()
  36. {
  37. QTest::addColumn<bool>("doLocalDiscovery");
  38. QTest::newRow("full local discovery") << true;
  39. QTest::newRow("skip local discovery") << false;
  40. }
  41. void testVirtualFileLifecycle()
  42. {
  43. QFETCH(bool, doLocalDiscovery);
  44. FakeFolder fakeFolder{ FileInfo() };
  45. SyncOptions syncOptions;
  46. syncOptions._newFilesAreVirtual = true;
  47. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  48. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  49. QSignalSpy completeSpy(&fakeFolder.syncEngine(), SIGNAL(itemCompleted(const SyncFileItemPtr &)));
  50. auto cleanup = [&]() {
  51. completeSpy.clear();
  52. if (!doLocalDiscovery)
  53. fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::DatabaseAndFilesystem);
  54. };
  55. cleanup();
  56. // Create a virtual file for a new remote file
  57. fakeFolder.remoteModifier().mkdir("A");
  58. fakeFolder.remoteModifier().insert("A/a1", 64);
  59. QVERIFY(fakeFolder.syncOnce());
  60. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  61. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  62. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  63. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_NEW));
  64. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFile);
  65. cleanup();
  66. // Another sync doesn't actually lead to changes
  67. QVERIFY(fakeFolder.syncOnce());
  68. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  69. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  70. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  71. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFile);
  72. QVERIFY(completeSpy.isEmpty());
  73. cleanup();
  74. // Not even when the remote is rediscovered
  75. fakeFolder.syncJournal().forceRemoteDiscoveryNextSync();
  76. QVERIFY(fakeFolder.syncOnce());
  77. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  78. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  79. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  80. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFile);
  81. QVERIFY(completeSpy.isEmpty());
  82. cleanup();
  83. // Neither does a remote change
  84. fakeFolder.remoteModifier().appendByte("A/a1");
  85. QVERIFY(fakeFolder.syncOnce());
  86. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  87. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  88. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  89. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_UPDATE_METADATA));
  90. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFile);
  91. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._fileSize, 65);
  92. cleanup();
  93. // If the local virtual file file is removed, it'll just be recreated
  94. if (!doLocalDiscovery)
  95. fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::DatabaseAndFilesystem, { "A" });
  96. fakeFolder.localModifier().remove("A/a1.owncloud");
  97. QVERIFY(fakeFolder.syncOnce());
  98. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  99. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  100. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  101. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_NEW));
  102. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFile);
  103. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._fileSize, 65);
  104. cleanup();
  105. // Remote rename is propagated
  106. fakeFolder.remoteModifier().rename("A/a1", "A/a1m");
  107. QVERIFY(fakeFolder.syncOnce());
  108. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  109. QVERIFY(!fakeFolder.currentLocalState().find("A/a1m"));
  110. QVERIFY(!fakeFolder.currentLocalState().find("A/a1.owncloud"));
  111. QVERIFY(fakeFolder.currentLocalState().find("A/a1m.owncloud"));
  112. QVERIFY(!fakeFolder.currentRemoteState().find("A/a1"));
  113. QVERIFY(fakeFolder.currentRemoteState().find("A/a1m"));
  114. QVERIFY(itemInstruction(completeSpy, "A/a1m.owncloud", CSYNC_INSTRUCTION_RENAME));
  115. QCOMPARE(dbRecord(fakeFolder, "A/a1m.owncloud")._type, ItemTypeVirtualFile);
  116. cleanup();
  117. // Remote remove is propagated
  118. fakeFolder.remoteModifier().remove("A/a1m");
  119. QVERIFY(fakeFolder.syncOnce());
  120. QVERIFY(!fakeFolder.currentLocalState().find("A/a1m.owncloud"));
  121. QVERIFY(!fakeFolder.currentRemoteState().find("A/a1m"));
  122. QVERIFY(itemInstruction(completeSpy, "A/a1m.owncloud", CSYNC_INSTRUCTION_REMOVE));
  123. QVERIFY(!dbRecord(fakeFolder, "A/a1.owncloud").isValid());
  124. QVERIFY(!dbRecord(fakeFolder, "A/a1m.owncloud").isValid());
  125. cleanup();
  126. // Edge case: Local virtual file but no db entry for some reason
  127. fakeFolder.remoteModifier().insert("A/a2", 64);
  128. fakeFolder.remoteModifier().insert("A/a3", 64);
  129. QVERIFY(fakeFolder.syncOnce());
  130. QVERIFY(fakeFolder.currentLocalState().find("A/a2.owncloud"));
  131. QVERIFY(fakeFolder.currentLocalState().find("A/a3.owncloud"));
  132. cleanup();
  133. fakeFolder.syncEngine().journal()->deleteFileRecord("A/a2.owncloud");
  134. fakeFolder.syncEngine().journal()->deleteFileRecord("A/a3.owncloud");
  135. fakeFolder.remoteModifier().remove("A/a3");
  136. fakeFolder.syncEngine().setLocalDiscoveryOptions(LocalDiscoveryStyle::FilesystemOnly);
  137. QVERIFY(fakeFolder.syncOnce());
  138. QVERIFY(fakeFolder.currentLocalState().find("A/a2.owncloud"));
  139. QVERIFY(itemInstruction(completeSpy, "A/a2.owncloud", CSYNC_INSTRUCTION_NEW));
  140. QVERIFY(dbRecord(fakeFolder, "A/a2.owncloud").isValid());
  141. QVERIFY(!fakeFolder.currentLocalState().find("A/a3.owncloud"));
  142. QVERIFY(!dbRecord(fakeFolder, "A/a3.owncloud").isValid());
  143. cleanup();
  144. }
  145. void testVirtualFileConflict()
  146. {
  147. FakeFolder fakeFolder{ FileInfo() };
  148. SyncOptions syncOptions;
  149. syncOptions._newFilesAreVirtual = true;
  150. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  151. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  152. QSignalSpy completeSpy(&fakeFolder.syncEngine(), SIGNAL(itemCompleted(const SyncFileItemPtr &)));
  153. auto cleanup = [&]() {
  154. completeSpy.clear();
  155. };
  156. cleanup();
  157. // Create a virtual file for a new remote file
  158. fakeFolder.remoteModifier().mkdir("A");
  159. fakeFolder.remoteModifier().insert("A/a1", 64);
  160. fakeFolder.remoteModifier().insert("A/a2", 64);
  161. fakeFolder.remoteModifier().mkdir("B");
  162. fakeFolder.remoteModifier().insert("B/b1", 64);
  163. fakeFolder.remoteModifier().insert("B/b2", 64);
  164. fakeFolder.remoteModifier().mkdir("C");
  165. fakeFolder.remoteModifier().insert("C/c1", 64);
  166. QVERIFY(fakeFolder.syncOnce());
  167. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  168. QVERIFY(fakeFolder.currentLocalState().find("B/b2.owncloud"));
  169. cleanup();
  170. // A: the correct file and a conflicting file are added, virtual files stay
  171. // B: same setup, but the virtual files are deleted by the user
  172. // C: user adds a *directory* locally
  173. fakeFolder.localModifier().insert("A/a1", 64);
  174. fakeFolder.localModifier().insert("A/a2", 30);
  175. fakeFolder.localModifier().insert("B/b1", 64);
  176. fakeFolder.localModifier().insert("B/b2", 30);
  177. fakeFolder.localModifier().remove("B/b1.owncloud");
  178. fakeFolder.localModifier().remove("B/b2.owncloud");
  179. fakeFolder.localModifier().mkdir("C/c1");
  180. fakeFolder.localModifier().insert("C/c1/foo");
  181. QVERIFY(fakeFolder.syncOnce());
  182. // Everything is CONFLICT since mtimes are different even for a1/b1
  183. QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_CONFLICT));
  184. QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_CONFLICT));
  185. QVERIFY(itemInstruction(completeSpy, "B/b1", CSYNC_INSTRUCTION_CONFLICT));
  186. QVERIFY(itemInstruction(completeSpy, "B/b2", CSYNC_INSTRUCTION_CONFLICT));
  187. QVERIFY(itemInstruction(completeSpy, "C/c1", CSYNC_INSTRUCTION_CONFLICT));
  188. // no virtual file files should remain
  189. QVERIFY(!fakeFolder.currentLocalState().find("A/a1.owncloud"));
  190. QVERIFY(!fakeFolder.currentLocalState().find("A/a2.owncloud"));
  191. QVERIFY(!fakeFolder.currentLocalState().find("B/b1.owncloud"));
  192. QVERIFY(!fakeFolder.currentLocalState().find("B/b2.owncloud"));
  193. QVERIFY(!fakeFolder.currentLocalState().find("C/c1.owncloud"));
  194. // conflict files should exist
  195. QCOMPARE(fakeFolder.syncJournal().conflictRecordPaths().size(), 3);
  196. // nothing should have the virtual file tag
  197. QCOMPARE(dbRecord(fakeFolder, "A/a1")._type, ItemTypeFile);
  198. QCOMPARE(dbRecord(fakeFolder, "A/a2")._type, ItemTypeFile);
  199. QCOMPARE(dbRecord(fakeFolder, "B/b1")._type, ItemTypeFile);
  200. QCOMPARE(dbRecord(fakeFolder, "B/b2")._type, ItemTypeFile);
  201. QCOMPARE(dbRecord(fakeFolder, "C/c1")._type, ItemTypeFile);
  202. QVERIFY(!dbRecord(fakeFolder, "A/a1.owncloud").isValid());
  203. QVERIFY(!dbRecord(fakeFolder, "A/a2.owncloud").isValid());
  204. QVERIFY(!dbRecord(fakeFolder, "B/b1.owncloud").isValid());
  205. QVERIFY(!dbRecord(fakeFolder, "B/b2.owncloud").isValid());
  206. QVERIFY(!dbRecord(fakeFolder, "C/c1.owncloud").isValid());
  207. cleanup();
  208. }
  209. void testWithNormalSync()
  210. {
  211. FakeFolder fakeFolder{ FileInfo::A12_B12_C12_S12() };
  212. SyncOptions syncOptions;
  213. syncOptions._newFilesAreVirtual = true;
  214. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  215. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  216. QSignalSpy completeSpy(&fakeFolder.syncEngine(), SIGNAL(itemCompleted(const SyncFileItemPtr &)));
  217. auto cleanup = [&]() {
  218. completeSpy.clear();
  219. };
  220. cleanup();
  221. // No effect sync
  222. QVERIFY(fakeFolder.syncOnce());
  223. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  224. cleanup();
  225. // Existing files are propagated just fine in both directions
  226. fakeFolder.localModifier().appendByte("A/a1");
  227. fakeFolder.localModifier().insert("A/a3");
  228. fakeFolder.remoteModifier().appendByte("A/a2");
  229. QVERIFY(fakeFolder.syncOnce());
  230. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  231. cleanup();
  232. // New files on the remote create virtual files
  233. fakeFolder.remoteModifier().insert("A/new");
  234. QVERIFY(fakeFolder.syncOnce());
  235. QVERIFY(!fakeFolder.currentLocalState().find("A/new"));
  236. QVERIFY(fakeFolder.currentLocalState().find("A/new.owncloud"));
  237. QVERIFY(fakeFolder.currentRemoteState().find("A/new"));
  238. QVERIFY(itemInstruction(completeSpy, "A/new.owncloud", CSYNC_INSTRUCTION_NEW));
  239. QCOMPARE(dbRecord(fakeFolder, "A/new.owncloud")._type, ItemTypeVirtualFile);
  240. cleanup();
  241. }
  242. void testVirtualFileDownload()
  243. {
  244. FakeFolder fakeFolder{ FileInfo() };
  245. SyncOptions syncOptions;
  246. syncOptions._newFilesAreVirtual = true;
  247. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  248. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  249. QSignalSpy completeSpy(&fakeFolder.syncEngine(), SIGNAL(itemCompleted(const SyncFileItemPtr &)));
  250. auto cleanup = [&]() {
  251. completeSpy.clear();
  252. };
  253. cleanup();
  254. auto triggerDownload = [&](const QByteArray &path) {
  255. auto &journal = fakeFolder.syncJournal();
  256. SyncJournalFileRecord record;
  257. journal.getFileRecord(path + ".owncloud", &record);
  258. if (!record.isValid())
  259. return;
  260. record._type = ItemTypeVirtualFileDownload;
  261. journal.setFileRecord(record);
  262. };
  263. // Create a virtual file for remote files
  264. fakeFolder.remoteModifier().mkdir("A");
  265. fakeFolder.remoteModifier().insert("A/a1");
  266. fakeFolder.remoteModifier().insert("A/a2");
  267. fakeFolder.remoteModifier().insert("A/a3");
  268. fakeFolder.remoteModifier().insert("A/a4");
  269. fakeFolder.remoteModifier().insert("A/a5");
  270. fakeFolder.remoteModifier().insert("A/a6");
  271. QVERIFY(fakeFolder.syncOnce());
  272. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  273. QVERIFY(fakeFolder.currentLocalState().find("A/a2.owncloud"));
  274. QVERIFY(fakeFolder.currentLocalState().find("A/a3.owncloud"));
  275. QVERIFY(fakeFolder.currentLocalState().find("A/a4.owncloud"));
  276. QVERIFY(fakeFolder.currentLocalState().find("A/a5.owncloud"));
  277. QVERIFY(fakeFolder.currentLocalState().find("A/a6.owncloud"));
  278. cleanup();
  279. // Download by changing the db entry
  280. triggerDownload("A/a1");
  281. triggerDownload("A/a2");
  282. triggerDownload("A/a3");
  283. triggerDownload("A/a4");
  284. triggerDownload("A/a5");
  285. triggerDownload("A/a6");
  286. fakeFolder.remoteModifier().appendByte("A/a2");
  287. fakeFolder.remoteModifier().remove("A/a3");
  288. fakeFolder.remoteModifier().rename("A/a4", "A/a4m");
  289. fakeFolder.localModifier().insert("A/a5");
  290. fakeFolder.localModifier().insert("A/a6");
  291. fakeFolder.localModifier().remove("A/a6.owncloud");
  292. QVERIFY(fakeFolder.syncOnce());
  293. QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_NEW));
  294. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_NONE));
  295. QVERIFY(itemInstruction(completeSpy, "A/a2", CSYNC_INSTRUCTION_NEW));
  296. QVERIFY(itemInstruction(completeSpy, "A/a2.owncloud", CSYNC_INSTRUCTION_NONE));
  297. QVERIFY(itemInstruction(completeSpy, "A/a3.owncloud", CSYNC_INSTRUCTION_REMOVE));
  298. QVERIFY(itemInstruction(completeSpy, "A/a4m", CSYNC_INSTRUCTION_NEW));
  299. QVERIFY(itemInstruction(completeSpy, "A/a4.owncloud", CSYNC_INSTRUCTION_REMOVE));
  300. QVERIFY(itemInstruction(completeSpy, "A/a5", CSYNC_INSTRUCTION_CONFLICT));
  301. QVERIFY(itemInstruction(completeSpy, "A/a5.owncloud", CSYNC_INSTRUCTION_NONE));
  302. QVERIFY(itemInstruction(completeSpy, "A/a6", CSYNC_INSTRUCTION_CONFLICT));
  303. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  304. QCOMPARE(dbRecord(fakeFolder, "A/a1")._type, ItemTypeFile);
  305. QCOMPARE(dbRecord(fakeFolder, "A/a2")._type, ItemTypeFile);
  306. QVERIFY(!dbRecord(fakeFolder, "A/a3").isValid());
  307. QCOMPARE(dbRecord(fakeFolder, "A/a4m")._type, ItemTypeFile);
  308. QCOMPARE(dbRecord(fakeFolder, "A/a5")._type, ItemTypeFile);
  309. QCOMPARE(dbRecord(fakeFolder, "A/a6")._type, ItemTypeFile);
  310. QVERIFY(!dbRecord(fakeFolder, "A/a1.owncloud").isValid());
  311. QVERIFY(!dbRecord(fakeFolder, "A/a2.owncloud").isValid());
  312. QVERIFY(!dbRecord(fakeFolder, "A/a3.owncloud").isValid());
  313. QVERIFY(!dbRecord(fakeFolder, "A/a4.owncloud").isValid());
  314. QVERIFY(!dbRecord(fakeFolder, "A/a5.owncloud").isValid());
  315. QVERIFY(!dbRecord(fakeFolder, "A/a6.owncloud").isValid());
  316. }
  317. void testVirtualFileDownloadResume()
  318. {
  319. FakeFolder fakeFolder{ FileInfo() };
  320. SyncOptions syncOptions;
  321. syncOptions._newFilesAreVirtual = true;
  322. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  323. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  324. QSignalSpy completeSpy(&fakeFolder.syncEngine(), SIGNAL(itemCompleted(const SyncFileItemPtr &)));
  325. auto cleanup = [&]() {
  326. completeSpy.clear();
  327. fakeFolder.syncJournal().wipeErrorBlacklist();
  328. };
  329. cleanup();
  330. auto triggerDownload = [&](const QByteArray &path) {
  331. auto &journal = fakeFolder.syncJournal();
  332. SyncJournalFileRecord record;
  333. journal.getFileRecord(path + ".owncloud", &record);
  334. if (!record.isValid())
  335. return;
  336. record._type = ItemTypeVirtualFileDownload;
  337. journal.setFileRecord(record);
  338. journal.avoidReadFromDbOnNextSync(record._path);
  339. };
  340. // Create a virtual file for remote files
  341. fakeFolder.remoteModifier().mkdir("A");
  342. fakeFolder.remoteModifier().insert("A/a1");
  343. QVERIFY(fakeFolder.syncOnce());
  344. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  345. cleanup();
  346. // Download by changing the db entry
  347. triggerDownload("A/a1");
  348. fakeFolder.serverErrorPaths().append("A/a1", 500);
  349. QVERIFY(!fakeFolder.syncOnce());
  350. QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_NEW));
  351. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_NONE));
  352. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  353. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  354. QCOMPARE(dbRecord(fakeFolder, "A/a1.owncloud")._type, ItemTypeVirtualFileDownload);
  355. QVERIFY(!dbRecord(fakeFolder, "A/a1").isValid());
  356. cleanup();
  357. fakeFolder.serverErrorPaths().clear();
  358. QVERIFY(fakeFolder.syncOnce());
  359. QVERIFY(itemInstruction(completeSpy, "A/a1", CSYNC_INSTRUCTION_NEW));
  360. QVERIFY(itemInstruction(completeSpy, "A/a1.owncloud", CSYNC_INSTRUCTION_NONE));
  361. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  362. QCOMPARE(dbRecord(fakeFolder, "A/a1")._type, ItemTypeFile);
  363. QVERIFY(!dbRecord(fakeFolder, "A/a1.owncloud").isValid());
  364. }
  365. // Check what might happen if an older sync client encounters virtual files
  366. void testOldVersion1()
  367. {
  368. FakeFolder fakeFolder{ FileInfo() };
  369. SyncOptions syncOptions;
  370. syncOptions._newFilesAreVirtual = true;
  371. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  372. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  373. // Create a virtual file
  374. fakeFolder.remoteModifier().mkdir("A");
  375. fakeFolder.remoteModifier().insert("A/a1");
  376. QVERIFY(fakeFolder.syncOnce());
  377. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  378. // Simulate an old client by switching the type of all ItemTypeVirtualFile
  379. // entries in the db to an invalid type.
  380. auto &db = fakeFolder.syncJournal();
  381. SyncJournalFileRecord rec;
  382. db.getFileRecord(QByteArray("A/a1.owncloud"), &rec);
  383. QVERIFY(rec.isValid());
  384. QCOMPARE(rec._type, ItemTypeVirtualFile);
  385. rec._type = static_cast<ItemType>(-1);
  386. db.setFileRecord(rec);
  387. // Also switch off new files becoming virtual files
  388. syncOptions._newFilesAreVirtual = false;
  389. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  390. // A sync that doesn't do remote discovery has no effect
  391. QVERIFY(fakeFolder.syncOnce());
  392. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  393. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  394. QVERIFY(fakeFolder.currentRemoteState().find("A/a1"));
  395. QVERIFY(!fakeFolder.currentRemoteState().find("A/a1.owncloud"));
  396. // But with a remote discovery the virtual files will be removed and
  397. // the remote files will be downloaded.
  398. db.forceRemoteDiscoveryNextSync();
  399. QVERIFY(fakeFolder.syncOnce());
  400. QVERIFY(fakeFolder.currentLocalState().find("A/a1"));
  401. QVERIFY(!fakeFolder.currentLocalState().find("A/a1.owncloud"));
  402. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  403. }
  404. // Older versions may leave db entries for foo and foo.owncloud
  405. void testOldVersion2()
  406. {
  407. FakeFolder fakeFolder{ FileInfo() };
  408. // Sync a file
  409. fakeFolder.remoteModifier().mkdir("A");
  410. fakeFolder.remoteModifier().insert("A/a1");
  411. QVERIFY(fakeFolder.syncOnce());
  412. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  413. // Create the virtual file too
  414. // In the wild, the new version would create the virtual file and the db entry
  415. // while the old version would download the plain file.
  416. fakeFolder.localModifier().insert("A/a1.owncloud");
  417. auto &db = fakeFolder.syncJournal();
  418. SyncJournalFileRecord rec;
  419. db.getFileRecord(QByteArray("A/a1"), &rec);
  420. rec._type = ItemTypeVirtualFile;
  421. rec._path = "A/a1.owncloud";
  422. db.setFileRecord(rec);
  423. SyncOptions syncOptions;
  424. syncOptions._newFilesAreVirtual = true;
  425. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  426. // Check that a sync removes the virtual file and its db entry
  427. QVERIFY(fakeFolder.syncOnce());
  428. QVERIFY(!fakeFolder.currentLocalState().find("A/a1.owncloud"));
  429. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  430. QVERIFY(!dbRecord(fakeFolder, "A/a1.owncloud").isValid());
  431. }
  432. void testDownloadRecursive()
  433. {
  434. FakeFolder fakeFolder{ FileInfo() };
  435. SyncOptions syncOptions;
  436. syncOptions._newFilesAreVirtual = true;
  437. fakeFolder.syncEngine().setSyncOptions(syncOptions);
  438. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  439. // Create a virtual file for remote files
  440. fakeFolder.remoteModifier().mkdir("A");
  441. fakeFolder.remoteModifier().mkdir("A/Sub");
  442. fakeFolder.remoteModifier().mkdir("A/Sub/SubSub");
  443. fakeFolder.remoteModifier().mkdir("A/Sub2");
  444. fakeFolder.remoteModifier().mkdir("B");
  445. fakeFolder.remoteModifier().mkdir("B/Sub");
  446. fakeFolder.remoteModifier().insert("A/a1");
  447. fakeFolder.remoteModifier().insert("A/a2");
  448. fakeFolder.remoteModifier().insert("A/Sub/a3");
  449. fakeFolder.remoteModifier().insert("A/Sub/a4");
  450. fakeFolder.remoteModifier().insert("A/Sub/SubSub/a5");
  451. fakeFolder.remoteModifier().insert("A/Sub2/a6");
  452. fakeFolder.remoteModifier().insert("B/b1");
  453. fakeFolder.remoteModifier().insert("B/Sub/b2");
  454. QVERIFY(fakeFolder.syncOnce());
  455. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  456. QVERIFY(fakeFolder.currentLocalState().find("A/a2.owncloud"));
  457. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a3.owncloud"));
  458. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a4.owncloud"));
  459. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/SubSub/a5.owncloud"));
  460. QVERIFY(fakeFolder.currentLocalState().find("A/Sub2/a6.owncloud"));
  461. QVERIFY(fakeFolder.currentLocalState().find("B/b1.owncloud"));
  462. QVERIFY(fakeFolder.currentLocalState().find("B/Sub/b2.owncloud"));
  463. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  464. QVERIFY(!fakeFolder.currentLocalState().find("A/a2"));
  465. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a3"));
  466. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a4"));
  467. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/SubSub/a5"));
  468. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub2/a6"));
  469. QVERIFY(!fakeFolder.currentLocalState().find("B/b1"));
  470. QVERIFY(!fakeFolder.currentLocalState().find("B/Sub/b2"));
  471. // Download All file in the directory A/Sub
  472. // (as in Folder::downloadVirtualFile)
  473. fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("A/Sub");
  474. QVERIFY(fakeFolder.syncOnce());
  475. QVERIFY(fakeFolder.currentLocalState().find("A/a1.owncloud"));
  476. QVERIFY(fakeFolder.currentLocalState().find("A/a2.owncloud"));
  477. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a3.owncloud"));
  478. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a4.owncloud"));
  479. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/SubSub/a5.owncloud"));
  480. QVERIFY(fakeFolder.currentLocalState().find("A/Sub2/a6.owncloud"));
  481. QVERIFY(fakeFolder.currentLocalState().find("B/b1.owncloud"));
  482. QVERIFY(fakeFolder.currentLocalState().find("B/Sub/b2.owncloud"));
  483. QVERIFY(!fakeFolder.currentLocalState().find("A/a1"));
  484. QVERIFY(!fakeFolder.currentLocalState().find("A/a2"));
  485. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a3"));
  486. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a4"));
  487. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/SubSub/a5"));
  488. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub2/a6"));
  489. QVERIFY(!fakeFolder.currentLocalState().find("B/b1"));
  490. QVERIFY(!fakeFolder.currentLocalState().find("B/Sub/b2"));
  491. // Add a file in a subfolder that was downloaded
  492. // Currently, this continue to add it as a virtual file.
  493. fakeFolder.remoteModifier().insert("A/Sub/SubSub/a7");
  494. QVERIFY(fakeFolder.syncOnce());
  495. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/SubSub/a7.owncloud"));
  496. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/SubSub/a7"));
  497. // Now download all files in "A"
  498. fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("A");
  499. QVERIFY(fakeFolder.syncOnce());
  500. QVERIFY(!fakeFolder.currentLocalState().find("A/a1.owncloud"));
  501. QVERIFY(!fakeFolder.currentLocalState().find("A/a2.owncloud"));
  502. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a3.owncloud"));
  503. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/a4.owncloud"));
  504. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/SubSub/a5.owncloud"));
  505. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub2/a6.owncloud"));
  506. QVERIFY(!fakeFolder.currentLocalState().find("A/Sub/SubSub/a7.owncloud"));
  507. QVERIFY(fakeFolder.currentLocalState().find("B/b1.owncloud"));
  508. QVERIFY(fakeFolder.currentLocalState().find("B/Sub/b2.owncloud"));
  509. QVERIFY(fakeFolder.currentLocalState().find("A/a1"));
  510. QVERIFY(fakeFolder.currentLocalState().find("A/a2"));
  511. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a3"));
  512. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/a4"));
  513. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/SubSub/a5"));
  514. QVERIFY(fakeFolder.currentLocalState().find("A/Sub2/a6"));
  515. QVERIFY(fakeFolder.currentLocalState().find("A/Sub/SubSub/a7"));
  516. QVERIFY(!fakeFolder.currentLocalState().find("B/b1"));
  517. QVERIFY(!fakeFolder.currentLocalState().find("B/Sub/b2"));
  518. // Now download remaining files in "B"
  519. fakeFolder.syncJournal().markVirtualFileForDownloadRecursively("B");
  520. QVERIFY(fakeFolder.syncOnce());
  521. QCOMPARE(fakeFolder.currentLocalState(), fakeFolder.currentRemoteState());
  522. }
  523. };
  524. QTEST_GUILESS_MAIN(TestSyncVirtualFiles)
  525. #include "testsyncvirtualfiles.moc"