050-file_iterator.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. // SPDX-FileCopyrightText: 2019-2024 Ivan Baidakou
  3. #include "test-utils.h"
  4. #include "model/cluster.h"
  5. #include "model/misc/file_iterator.h"
  6. #include "model/misc/sequencer.h"
  7. #include "diff-builder.h"
  8. #include <set>
  9. using namespace syncspirit;
  10. using namespace syncspirit::test;
  11. using namespace syncspirit::model;
  12. struct dummy_sink_t final : model::diff_sink_t {
  13. dummy_sink_t(diff_builder_t &builder_) noexcept : builder{builder_} {}
  14. void push(diff::cluster_diff_ptr_t d) noexcept override { builder.assign(d.get()); }
  15. diff_builder_t &builder;
  16. };
  17. TEST_CASE("file iterator, single folder", "[model]") {
  18. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  19. auto my_device = device_t::create(my_id, "my-device").value();
  20. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  21. auto peer_device = device_t::create(peer_id, "peer-device").value();
  22. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1));
  23. auto sequencer = make_sequencer(4);
  24. cluster->get_devices().put(my_device);
  25. cluster->get_devices().put(peer_device);
  26. auto builder = diff_builder_t(*cluster);
  27. auto sink = dummy_sink_t(builder);
  28. auto &blocks_map = cluster->get_blocks();
  29. auto &folders = cluster->get_folders();
  30. REQUIRE(builder.upsert_folder("1234-5678", "/my/path").apply());
  31. REQUIRE(builder.share_folder(peer_id.get_sha256(), "1234-5678").apply());
  32. auto folder = folders.by_id("1234-5678");
  33. auto &folder_infos = cluster->get_folders().by_id(folder->get_id())->get_folder_infos();
  34. REQUIRE(folder_infos.size() == 2u);
  35. auto peer_folder = folder_infos.by_device(*peer_device);
  36. auto &peer_files = peer_folder->get_file_infos();
  37. auto my_folder = folder_infos.by_device(*my_device);
  38. auto &my_files = my_folder->get_file_infos();
  39. auto file_iterator = peer_device->create_iterator(*cluster);
  40. file_iterator->activate(sink);
  41. SECTION("emtpy folders (1)") {
  42. CHECK(!file_iterator->next_need_cloning());
  43. CHECK(!file_iterator->next_need_sync());
  44. }
  45. REQUIRE(builder.configure_cluster(peer_id.get_sha256())
  46. .add(peer_id.get_sha256(), folder->get_id(), 123, 10u)
  47. .finish()
  48. .apply());
  49. SECTION("emtpy folders (2)") {
  50. CHECK(!file_iterator->next_need_cloning());
  51. CHECK(!file_iterator->next_need_sync());
  52. }
  53. SECTION("cloning (empty files)") {
  54. SECTION("1 file") {
  55. SECTION("no local file") {
  56. auto file = proto::FileInfo();
  57. file.set_name("a.txt");
  58. file.set_sequence(10ul);
  59. REQUIRE(
  60. builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish().apply());
  61. auto f = file_iterator->next_need_cloning();
  62. REQUIRE(f);
  63. CHECK(f->get_name() == "a.txt");
  64. CHECK(!f->is_locked());
  65. REQUIRE(builder.apply());
  66. CHECK(folder->is_synchronizing());
  67. REQUIRE(!file_iterator->next_need_cloning());
  68. REQUIRE(!file_iterator->next_need_sync());
  69. REQUIRE(builder.clone_file(*f).apply());
  70. REQUIRE(!file_iterator->next_need_cloning());
  71. REQUIRE(!file_iterator->next_need_sync());
  72. CHECK(!folder->is_synchronizing());
  73. CHECK(!f->is_locked());
  74. }
  75. SECTION("invalid file is ignored") {
  76. auto file = proto::FileInfo();
  77. file.set_name("a.txt");
  78. file.set_sequence(10ul);
  79. file.set_invalid(true);
  80. REQUIRE(builder.apply());
  81. REQUIRE(!file_iterator->next_need_cloning());
  82. }
  83. SECTION("version checks") {
  84. auto file = proto::FileInfo();
  85. file.set_name("a.txt");
  86. file.set_sequence(10ul);
  87. auto c_1 = file.mutable_version()->add_counters();
  88. c_1->set_id(1);
  89. c_1->set_value(5);
  90. builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish();
  91. SECTION("my version < peer version") {
  92. c_1->set_value(3);
  93. REQUIRE(builder.apply());
  94. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  95. my_file->mark_local();
  96. my_files.put(my_file);
  97. auto f = file_iterator->next_need_cloning();
  98. REQUIRE(f);
  99. CHECK(f->get_folder_info()->get_device() == peer_device.get());
  100. REQUIRE(!file_iterator->next_need_cloning());
  101. }
  102. SECTION("my version < peer version, but not scanned yet") {
  103. REQUIRE(builder.apply());
  104. c_1->set_value(3);
  105. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  106. my_files.put(my_file);
  107. REQUIRE(!file_iterator->next_need_cloning());
  108. }
  109. SECTION("my version > peer version") {
  110. REQUIRE(builder.apply());
  111. c_1->set_value(10);
  112. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  113. my_file->mark_local();
  114. my_files.put(my_file);
  115. REQUIRE(!file_iterator->next_need_cloning());
  116. }
  117. SECTION("my version == peer version") {
  118. REQUIRE(builder.apply());
  119. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  120. my_file->mark_local();
  121. my_files.put(my_file);
  122. REQUIRE(!file_iterator->next_need_cloning());
  123. }
  124. }
  125. }
  126. SECTION("2 files") {
  127. auto files = model::file_infos_map_t();
  128. auto file_1 = proto::FileInfo();
  129. file_1.set_name("a.txt");
  130. file_1.set_sequence(10ul);
  131. auto file_2 = proto::FileInfo();
  132. file_2.set_name("b.txt");
  133. file_2.set_sequence(10ul);
  134. builder.make_index(peer_id.get_sha256(), folder->get_id())
  135. .add(file_1, peer_device)
  136. .add(file_2, peer_device)
  137. .finish();
  138. REQUIRE(builder.apply());
  139. SECTION("both files are missing on my side") {
  140. auto f = file_iterator->next_need_cloning();
  141. REQUIRE(f);
  142. CHECK((f->get_name() == "a.txt" || f->get_name() == "b.txt"));
  143. CHECK(!f->is_locked());
  144. files.put(f);
  145. REQUIRE(builder.apply());
  146. CHECK(folder->is_synchronizing());
  147. REQUIRE(builder.clone_file(*f).apply());
  148. f = file_iterator->next_need_cloning();
  149. REQUIRE(f);
  150. CHECK((f->get_name() == "a.txt" || f->get_name() == "b.txt"));
  151. CHECK(!f->is_locked());
  152. files.put(f);
  153. REQUIRE(builder.apply());
  154. CHECK(folder->is_synchronizing());
  155. REQUIRE(builder.clone_file(*f).apply());
  156. REQUIRE(!file_iterator->next_need_cloning());
  157. REQUIRE(!file_iterator->next_need_sync());
  158. CHECK(!folder->is_synchronizing());
  159. CHECK(files.by_name("a.txt"));
  160. CHECK(files.by_name("b.txt"));
  161. }
  162. SECTION("1 file is missing on my side") {
  163. auto peer_file = peer_files.by_name("a.txt");
  164. REQUIRE(peer_file);
  165. REQUIRE(builder.clone_file(*peer_file).apply());
  166. auto f = file_iterator->next_need_cloning();
  167. REQUIRE(f);
  168. CHECK(f->get_name() == "b.txt");
  169. CHECK(!f->is_locked());
  170. REQUIRE(builder.apply());
  171. CHECK(folder->is_synchronizing());
  172. REQUIRE(!file_iterator->next_need_cloning());
  173. REQUIRE(builder.clone_file(*f).apply());
  174. REQUIRE(!file_iterator->next_need_cloning());
  175. REQUIRE(!file_iterator->next_need_sync());
  176. CHECK(!folder->is_synchronizing());
  177. CHECK(!f->is_locked());
  178. }
  179. SECTION("0 files are missing on my side") {
  180. auto peer_file_1 = peer_files.by_name("a.txt");
  181. auto peer_file_2 = peer_files.by_name("b.txt");
  182. REQUIRE(builder.clone_file(*peer_file_1).clone_file(*peer_file_2).apply());
  183. REQUIRE(!file_iterator->next_need_cloning());
  184. }
  185. }
  186. }
  187. SECTION("synchronizing non-empty files") {
  188. auto b1 = proto::BlockInfo();
  189. b1.set_hash(utils::sha256_digest("12345").value());
  190. b1.set_weak_hash(555);
  191. b1.set_size(5ul);
  192. auto b2 = proto::BlockInfo();
  193. b1.set_hash(utils::sha256_digest("67890").value());
  194. b1.set_weak_hash(123);
  195. b1.set_size(5ul);
  196. auto &blocks_map = cluster->get_blocks();
  197. blocks_map.put(block_info_t::create(b1).value());
  198. blocks_map.put(block_info_t::create(b2).value());
  199. SECTION("missing locally") {
  200. auto file = proto::FileInfo();
  201. file.set_name("a.txt");
  202. file.set_sequence(10ul);
  203. file.set_size(b1.size());
  204. file.set_block_size(b1.size());
  205. *file.add_blocks() = b1;
  206. REQUIRE(builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish().apply());
  207. auto f = file_iterator->next_need_sync();
  208. CHECK(!file_iterator->next_need_sync());
  209. REQUIRE(f);
  210. CHECK(f->get_name() == "a.txt");
  211. CHECK(!f->is_locked());
  212. REQUIRE(builder.apply());
  213. file_iterator->commit_sync(f);
  214. CHECK(!file_iterator->next_need_sync());
  215. }
  216. SECTION("have local, but outdated") {
  217. auto file = proto::FileInfo();
  218. file.set_name("a.txt");
  219. file.set_sequence(10ul);
  220. file.set_size(b1.size());
  221. file.set_block_size(b1.size());
  222. *file.add_blocks() = b1;
  223. auto c_1 = file.mutable_version()->add_counters();
  224. c_1->set_id(1);
  225. c_1->set_value(1);
  226. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  227. my_files.put(my_file);
  228. auto c_2 = file.mutable_version()->add_counters();
  229. c_2->set_id(2);
  230. c_2->set_value(2);
  231. REQUIRE(builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish().apply());
  232. SECTION("has been scanned") {
  233. my_file->mark_local();
  234. auto f = file_iterator->next_need_sync();
  235. REQUIRE(f);
  236. CHECK(f->get_name() == "a.txt");
  237. REQUIRE(file_iterator->next_need_sync() == f);
  238. REQUIRE(file_iterator->next_need_sync() == f);
  239. REQUIRE(builder.apply());
  240. file_iterator->commit_sync(f);
  241. CHECK(!file_iterator->next_need_sync());
  242. }
  243. SECTION("has not bee scanned") { CHECK(!file_iterator->next_need_sync()); }
  244. }
  245. SECTION("have local, local is newer") {
  246. auto file = proto::FileInfo();
  247. file.set_name("a.txt");
  248. file.set_sequence(10ul);
  249. file.set_size(b1.size());
  250. file.set_block_size(b1.size());
  251. *file.add_blocks() = b1;
  252. auto c_1 = file.mutable_version()->add_counters();
  253. c_1->set_id(1);
  254. c_1->set_value(1);
  255. builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish();
  256. auto c_2 = file.mutable_version()->add_counters();
  257. c_2->set_id(2);
  258. c_2->set_value(2);
  259. auto my_file = file_info_t::create(sequencer->next_uuid(), file, my_folder).value();
  260. my_files.put(my_file);
  261. REQUIRE(builder.apply());
  262. CHECK(!file_iterator->next_need_sync());
  263. }
  264. SECTION("peer file is unreacheable") {
  265. auto file = proto::FileInfo();
  266. file.set_name("a.txt");
  267. file.set_sequence(10ul);
  268. file.set_size(b1.size());
  269. file.set_block_size(b1.size());
  270. *file.add_blocks() = b1;
  271. REQUIRE(builder.make_index(peer_id.get_sha256(), folder->get_id()).add(file, peer_device).finish().apply());
  272. auto f = peer_files.by_name(file.name());
  273. REQUIRE(builder.clone_file(*f).mark_reacheable(f, false).apply());
  274. CHECK(!file_iterator->next_need_sync());
  275. }
  276. }
  277. file_iterator->deactivate();
  278. }
  279. TEST_CASE("file iterator for 2 folders", "[model]") {
  280. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  281. auto my_device = device_t::create(my_id, "my-device").value();
  282. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  283. auto peer_device = device_t::create(peer_id, "peer-device").value();
  284. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1));
  285. auto sequencer = make_sequencer(4);
  286. cluster->get_devices().put(my_device);
  287. cluster->get_devices().put(peer_device);
  288. auto builder = diff_builder_t(*cluster);
  289. auto sink = dummy_sink_t(builder);
  290. auto &blocks_map = cluster->get_blocks();
  291. auto &folders = cluster->get_folders();
  292. REQUIRE(builder.upsert_folder("1234-5678", "/my/path").apply());
  293. REQUIRE(builder.share_folder(peer_id.get_sha256(), "1234-5678").apply());
  294. auto folder = folders.by_id("1234-5678");
  295. auto &folder_infos = cluster->get_folders().by_id(folder->get_id())->get_folder_infos();
  296. REQUIRE(folder_infos.size() == 2u);
  297. auto peer_folder = folder_infos.by_device(*peer_device);
  298. auto &peer_files = peer_folder->get_file_infos();
  299. auto my_folder = folder_infos.by_device(*my_device);
  300. auto &my_files = my_folder->get_file_infos();
  301. auto file_iterator = peer_device->create_iterator(*cluster);
  302. file_iterator->activate(sink);
  303. auto sha256 = peer_id.get_sha256();
  304. REQUIRE(builder.upsert_folder("1234", "/", "my-label-1").upsert_folder("5678", "/", "my-label-2").apply());
  305. REQUIRE(builder.share_folder(sha256, "1234").share_folder(sha256, "5678").apply());
  306. auto folder1 = folders.by_id("1234");
  307. auto folder2 = folders.by_id("5678");
  308. REQUIRE(builder.configure_cluster(sha256)
  309. .add(sha256, "1234", 123, 10u)
  310. .add(sha256, "5678", 1234u, 11)
  311. .finish()
  312. .apply());
  313. auto file1 = proto::FileInfo();
  314. file1.set_name("a.txt");
  315. file1.set_sequence(10ul);
  316. auto file2 = proto::FileInfo();
  317. file2.set_name("b.txt");
  318. file2.set_sequence(11ul);
  319. using set_t = std::set<std::string_view>;
  320. SECTION("cloning") {
  321. REQUIRE(builder.make_index(sha256, "1234").add(file1, peer_device).add(file2, peer_device).finish().apply());
  322. auto f1 = file_iterator->next_need_cloning();
  323. auto f2 = file_iterator->next_need_cloning();
  324. REQUIRE(f1);
  325. REQUIRE(f2);
  326. auto files = set_t{};
  327. files.emplace(f1->get_name());
  328. files.emplace(f2->get_name());
  329. CHECK((files == set_t{"a.txt", "b.txt"}));
  330. CHECK(!file_iterator->next_need_cloning());
  331. }
  332. SECTION("syncing") {
  333. auto b1 = proto::BlockInfo();
  334. b1.set_hash(utils::sha256_digest("12345").value());
  335. b1.set_weak_hash(555);
  336. b1.set_size(5ul);
  337. auto b2 = proto::BlockInfo();
  338. b2.set_hash(utils::sha256_digest("67890").value());
  339. b2.set_weak_hash(123);
  340. b2.set_size(5ul);
  341. auto &blocks_map = cluster->get_blocks();
  342. blocks_map.put(block_info_t::create(b1).value());
  343. blocks_map.put(block_info_t::create(b2).value());
  344. *file1.add_blocks() = b1;
  345. file1.set_size(b1.size());
  346. *file2.add_blocks() = b2;
  347. file2.set_size(b2.size());
  348. using set_t = std::set<std::string_view>;
  349. REQUIRE(builder.make_index(sha256, "1234").add(file1, peer_device).add(file2, peer_device).finish().apply());
  350. auto files = set_t{};
  351. auto f1 = file_iterator->next_need_sync();
  352. REQUIRE(f1);
  353. files.emplace(f1->get_name());
  354. file_iterator->commit_sync(f1);
  355. auto f2 = file_iterator->next_need_sync();
  356. REQUIRE(f2);
  357. files.emplace(f2->get_name());
  358. file_iterator->commit_sync(f2);
  359. files.emplace(f1->get_name());
  360. files.emplace(f2->get_name());
  361. CHECK((files == set_t{"a.txt", "b.txt"}));
  362. }
  363. file_iterator->deactivate();
  364. }
  365. int _init() {
  366. utils::set_default("trace");
  367. return 1;
  368. }
  369. static int v = _init();