051-block_iterator.cpp 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. // SPDX-FileCopyrightText: 2019-2023 Ivan Baidakou
  3. #include "test-utils.h"
  4. #include "model/cluster.h"
  5. #include "model/misc/block_iterator.h"
  6. #include "diff-builder.h"
  7. using namespace syncspirit;
  8. using namespace syncspirit::test;
  9. using namespace syncspirit::model;
  10. TEST_CASE("block iterator", "[model]") {
  11. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  12. auto my_device = device_t::create(my_id, "my-device").value();
  13. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  14. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  15. cluster->get_devices().put(my_device);
  16. auto block_iterator = block_iterator_ptr_t();
  17. auto next = [&](file_info_ptr_t source, bool reset = false) -> file_block_t {
  18. if (source && source->is_file() && !source->is_deleted()) {
  19. if (reset) {
  20. block_iterator = new blocks_iterator_t(*source);
  21. }
  22. if (block_iterator && *block_iterator) {
  23. return block_iterator->next(!reset);
  24. }
  25. }
  26. return {};
  27. };
  28. auto &folders = cluster->get_folders();
  29. auto builder = diff_builder_t(*cluster);
  30. builder.create_folder("1234-5678", "some/path", "my-label");
  31. REQUIRE(builder.apply());
  32. auto folder = folders.by_id("1234-5678");
  33. auto &folder_infos = folder->get_folder_infos();
  34. auto my_folder = folder_infos.by_device(*my_device);
  35. auto p_file = proto::FileInfo();
  36. p_file.set_name("a.txt");
  37. p_file.set_sequence(2ul);
  38. SECTION("no blocks") {
  39. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  40. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  41. REQUIRE(!next(my_file, true));
  42. }
  43. auto b1_hash = utils::sha256_digest("12345").value();
  44. auto b2_hash = utils::sha256_digest("567890").value();
  45. auto b1 = p_file.add_blocks();
  46. b1->set_size(5);
  47. b1->set_weak_hash(12);
  48. b1->set_hash(b1_hash);
  49. SECTION("two blocks") {
  50. auto b2 = p_file.add_blocks();
  51. b2->set_size(5);
  52. b2->set_weak_hash(12);
  53. b2->set_hash(b2_hash);
  54. p_file.set_size(10ul);
  55. p_file.set_block_size(5ul);
  56. SECTION("no iteration upon deleted file") {
  57. p_file.set_deleted(true);
  58. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  59. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  60. CHECK(!next(my_file, true));
  61. }
  62. SECTION("normal iteration") {
  63. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  64. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  65. auto bi1 = cluster->get_blocks().get(b1_hash);
  66. auto bi2 = cluster->get_blocks().get(b2_hash);
  67. my_file->remove_blocks();
  68. my_file->assign_block(bi1, 0);
  69. my_file->assign_block(bi2, 1);
  70. auto fb1 = next(my_file, true);
  71. REQUIRE(fb1);
  72. CHECK(fb1.block()->get_hash() == b1_hash);
  73. CHECK(fb1.block_index() == 0);
  74. CHECK(fb1.file() == my_file.get());
  75. auto fb1_1 = next(my_file);
  76. REQUIRE(fb1_1);
  77. CHECK(fb1_1.block()->get_hash() == b1_hash);
  78. CHECK(fb1_1.block_index() == 0);
  79. CHECK(fb1_1.file() == my_file.get());
  80. auto fb2 = next(my_file);
  81. REQUIRE(fb2);
  82. CHECK(fb2.block()->get_hash() == b2_hash);
  83. CHECK(fb2.block_index() == 1);
  84. CHECK(fb2.file() == my_file.get());
  85. REQUIRE(!next(my_file));
  86. }
  87. SECTION("no iteration upon unavailable file") {
  88. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  89. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  90. auto bi1 = cluster->get_blocks().get(b1_hash);
  91. auto bi2 = cluster->get_blocks().get(b2_hash);
  92. my_file->remove_blocks();
  93. my_file->assign_block(bi1, 0);
  94. my_file->assign_block(bi2, 1);
  95. my_file->mark_unreachable(true);
  96. CHECK(!next(my_file, true));
  97. }
  98. }
  99. SECTION("block is available on some other local file") {
  100. auto p_file_2 = proto::FileInfo();
  101. p_file_2.set_name("b.txt");
  102. p_file_2.set_sequence(2ul);
  103. p_file_2.set_size(b1->size());
  104. *p_file_2.add_blocks() = *b1;
  105. p_file.set_size(b1->size());
  106. REQUIRE(builder.local_update(folder->get_id(), p_file_2).apply());
  107. auto my_file_2 = my_folder->get_file_infos().by_name(p_file_2.name());
  108. REQUIRE(my_file_2->is_locally_available());
  109. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  110. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  111. auto bi1 = cluster->get_blocks().get(b1_hash);
  112. my_file->remove_blocks();
  113. my_file->assign_block(bi1, 0);
  114. REQUIRE(!my_file->is_locally_available());
  115. auto fb1 = next(my_file, true);
  116. REQUIRE(fb1);
  117. CHECK(fb1.block()->get_hash() == b1_hash);
  118. CHECK(fb1.block_index() == 0);
  119. CHECK(fb1.file() == my_file.get());
  120. }
  121. SECTION("locked/unlock blocks") {
  122. p_file.set_size(5ul);
  123. p_file.set_block_size(5ul);
  124. REQUIRE(builder.local_update(folder->get_id(), p_file).apply());
  125. auto my_file = my_folder->get_file_infos().by_name(p_file.name());
  126. auto bi1 = cluster->get_blocks().get(b1_hash);
  127. my_file->remove_blocks();
  128. my_file->assign_block(bi1, 0);
  129. REQUIRE(!my_file->is_locally_available());
  130. auto fb = next(my_file, true);
  131. REQUIRE(fb);
  132. auto block = fb.block();
  133. block->lock();
  134. REQUIRE(!next(my_file, true));
  135. block->unlock();
  136. REQUIRE(next(my_file, true));
  137. }
  138. }