031-diff-cluster_update.cpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692
  1. // SPDX-License-Identifier: GPL-3.0-or-later
  2. // SPDX-FileCopyrightText: 2019-2024 Ivan Baidakou
  3. #include "test-utils.h"
  4. #include "access.h"
  5. #include "model/cluster.h"
  6. #include "model/diff/peer/cluster_update.h"
  7. #include "model/diff/cluster_visitor.h"
  8. using namespace syncspirit;
  9. using namespace syncspirit::model;
  10. using namespace syncspirit::proto;
  11. using namespace syncspirit::test;
  12. template <typename F> struct my_cluster_update_visitor_t : diff::cluster_visitor_t {
  13. F fn;
  14. int remove_blocks = 0;
  15. int remove_files = 0;
  16. int remove_folders = 0;
  17. int remove_unknown_folders = 0;
  18. int add_unknown_folders = 0;
  19. int updated_folders = 0;
  20. my_cluster_update_visitor_t(F &&fn_) : fn{std::forward<F>(fn_)} {}
  21. outcome::result<void> operator()(const diff::peer::cluster_update_t &diff, void *custom) noexcept override {
  22. std::ignore = diff.diff::cluster_aggregate_diff_t::visit(*this, custom);
  23. return fn(diff);
  24. }
  25. outcome::result<void> operator()(const diff::modify::remove_blocks_t &, void *) noexcept override {
  26. ++remove_blocks;
  27. return outcome::success();
  28. }
  29. outcome::result<void> operator()(const diff::modify::remove_files_t &, void *) noexcept override {
  30. ++remove_files;
  31. return outcome::success();
  32. }
  33. outcome::result<void> operator()(const diff::modify::remove_folder_infos_t &, void *) noexcept override {
  34. ++remove_folders;
  35. return outcome::success();
  36. }
  37. outcome::result<void> operator()(const diff::modify::remove_unknown_folders_t &, void *) noexcept override {
  38. ++remove_unknown_folders;
  39. return outcome::success();
  40. }
  41. outcome::result<void> operator()(const diff::modify::update_folder_info_t &, void *) noexcept override {
  42. ++updated_folders;
  43. return outcome::success();
  44. }
  45. outcome::result<void> operator()(const diff::modify::add_unknown_folders_t &, void *) noexcept override {
  46. ++add_unknown_folders;
  47. return outcome::success();
  48. }
  49. };
  50. TEST_CASE("cluster update, new folder", "[model]") {
  51. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  52. auto my_device = device_t::create(my_id, "my-device").value();
  53. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  54. auto peer_device = device_t::create(peer_id, "peer-device").value();
  55. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  56. cluster->get_devices().put(my_device);
  57. cluster->get_devices().put(peer_device);
  58. SECTION("unknown folder") {
  59. auto cc = std::make_unique<proto::ClusterConfig>();
  60. auto folder_ptr = cc->add_folders();
  61. folder_ptr->set_id("some-id");
  62. folder_ptr->set_label("some-label");
  63. auto d_peer = folder_ptr->add_devices();
  64. d_peer->set_id(std::string(peer_id.get_sha256()));
  65. d_peer->set_name(std::string(peer_device->get_name()));
  66. d_peer->set_max_sequence(10);
  67. d_peer->set_index_id(22ul);
  68. auto folder = *folder_ptr;
  69. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  70. REQUIRE(diff_opt);
  71. auto &diff = diff_opt.value();
  72. auto r_a = diff->apply(*cluster);
  73. CHECK(r_a);
  74. auto visitor = my_cluster_update_visitor_t([&](auto &diff) { return outcome::success(); });
  75. auto r_v = diff->visit(visitor, nullptr);
  76. REQUIRE(r_v);
  77. REQUIRE(cluster->get_unknown_folders().size());
  78. REQUIRE(visitor.add_unknown_folders == 1);
  79. auto uf = cluster->get_unknown_folders().begin()->item;
  80. CHECK(uf->device_id() == peer_device->device_id());
  81. CHECK(uf->get_id() == "some-id");
  82. CHECK(uf->get_max_sequence() == 10);
  83. CHECK(uf->get_index() == 22ul);
  84. // no changes
  85. db::UnknownFolder db_uf;
  86. auto mf = db_uf.mutable_folder();
  87. mf->set_id(folder.id());
  88. auto mfi = db_uf.mutable_folder_info();
  89. mfi->set_max_sequence(d_peer->max_sequence());
  90. mfi->set_index_id(d_peer->index_id());
  91. diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  92. REQUIRE(diff_opt);
  93. diff = diff_opt.value();
  94. r_a = diff->apply(*cluster);
  95. CHECK(r_a);
  96. REQUIRE(cluster->get_unknown_folders().size() == 1);
  97. std::ignore = diff->visit(visitor, nullptr);
  98. REQUIRE(visitor.add_unknown_folders == 1);
  99. // max-id changed
  100. d_peer->set_max_sequence(15);
  101. diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  102. REQUIRE(diff_opt);
  103. diff = diff_opt.value();
  104. r_a = diff->apply(*cluster);
  105. CHECK(r_a);
  106. (void)diff->visit(visitor, nullptr);
  107. REQUIRE(visitor.add_unknown_folders == 2);
  108. REQUIRE(cluster->get_unknown_folders().size() == 1);
  109. uf = cluster->get_unknown_folders().begin()->item;
  110. CHECK(uf->device_id() == peer_device->device_id());
  111. CHECK(uf->get_id() == "some-id");
  112. CHECK(uf->get_max_sequence() == 15);
  113. CHECK(uf->get_index() == 22ul);
  114. CHECK(std::distance(cluster->get_unknown_folders().begin(), cluster->get_unknown_folders().end()) == 1);
  115. // change unknown folder
  116. cc = std::make_unique<proto::ClusterConfig>();
  117. folder_ptr = cc->add_folders();
  118. folder_ptr->set_id("some-id-n");
  119. folder_ptr->set_label("some-label-N");
  120. d_peer = folder_ptr->add_devices();
  121. d_peer->set_id(std::string(peer_id.get_sha256()));
  122. d_peer->set_name(std::string(peer_device->get_name()));
  123. d_peer->set_max_sequence(10);
  124. d_peer->set_index_id(22ul);
  125. diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  126. REQUIRE(diff_opt);
  127. r_a = diff_opt.value()->apply(*cluster);
  128. CHECK(r_a);
  129. CHECK(std::distance(cluster->get_unknown_folders().begin(), cluster->get_unknown_folders().end()) == 1);
  130. }
  131. SECTION("existing folder") {
  132. db::Folder db_folder;
  133. db_folder.set_id("1234-5678");
  134. db_folder.set_label("my-label");
  135. db_folder.set_path("/my/path");
  136. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  137. cluster->get_folders().put(folder);
  138. auto folder_info_my = folder_info_ptr_t();
  139. auto folder_info_peer = folder_info_ptr_t();
  140. {
  141. db::FolderInfo db_fi;
  142. db_fi.set_index_id(5ul);
  143. db_fi.set_max_sequence(10l);
  144. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  145. folder->get_folder_infos().put(folder_info_my);
  146. }
  147. {
  148. db::FolderInfo db_fi;
  149. db_fi.set_index_id(6ul);
  150. db_fi.set_max_sequence(10l);
  151. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  152. folder->get_folder_infos().put(folder_info_peer);
  153. }
  154. auto cc = std::make_unique<proto::ClusterConfig>();
  155. auto p_folder = cc->add_folders();
  156. p_folder->set_id(std::string(folder->get_id()));
  157. p_folder->set_label(std::string(folder->get_label()));
  158. auto p_peer = p_folder->add_devices();
  159. p_peer->set_max_sequence(folder_info_peer->get_max_sequence());
  160. p_peer->set_id(std::string(peer_id.get_sha256()));
  161. p_peer->set_name(std::string(peer_device->get_name()));
  162. SECTION("nothing changed") {
  163. p_peer->set_max_sequence(folder_info_peer->get_max_sequence());
  164. p_peer->set_index_id(folder_info_peer->get_index());
  165. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  166. REQUIRE(diff_opt);
  167. auto &diff = diff_opt.value();
  168. auto r_a = diff->apply(*cluster);
  169. CHECK(r_a);
  170. CHECK(!folder_info_peer->is_actual());
  171. auto pr_file = proto::FileInfo();
  172. pr_file.set_sequence(folder_info_peer->get_max_sequence());
  173. auto peer_file = file_info_t::create(cluster->next_uuid(), pr_file, folder_info_peer).value();
  174. folder_info_peer->add(peer_file, false);
  175. CHECK(folder_info_peer->is_actual());
  176. bool visited = false;
  177. auto visitor = my_cluster_update_visitor_t([&](auto &diff) {
  178. visited = true;
  179. return outcome::success();
  180. });
  181. auto r_v = diff->visit(visitor, nullptr);
  182. REQUIRE(r_v);
  183. REQUIRE(visited);
  184. REQUIRE(visitor.updated_folders == 0);
  185. }
  186. SECTION("max sequence increased") {
  187. auto max_seq = folder_info_peer->get_max_sequence() + 1;
  188. p_peer->set_max_sequence(max_seq);
  189. p_peer->set_index_id(folder_info_peer->get_index());
  190. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  191. REQUIRE(diff_opt);
  192. auto &diff = diff_opt.value();
  193. auto r_a = diff->apply(*cluster);
  194. CHECK(!folder_info_peer->is_actual());
  195. CHECK(r_a);
  196. bool visited = false;
  197. auto visitor = my_cluster_update_visitor_t([&](auto &diff) {
  198. visited = true;
  199. return outcome::success();
  200. });
  201. auto r_v = diff->visit(visitor, nullptr);
  202. REQUIRE(r_v);
  203. REQUIRE(visited);
  204. REQUIRE(visitor.updated_folders == 1);
  205. REQUIRE(folder_info_peer->get_max_sequence() == max_seq);
  206. }
  207. }
  208. SECTION("reset index") {
  209. db::Folder db_folder;
  210. db_folder.set_id("1234-5678");
  211. db_folder.set_label("my-label");
  212. db_folder.set_path("/my/path");
  213. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  214. cluster->get_folders().put(folder);
  215. auto folder_info_my = folder_info_ptr_t();
  216. auto folder_info_peer = folder_info_ptr_t();
  217. {
  218. db::FolderInfo db_fi;
  219. db_fi.set_index_id(5ul);
  220. db_fi.set_max_sequence(10l);
  221. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  222. folder->get_folder_infos().put(folder_info_my);
  223. }
  224. {
  225. db::FolderInfo db_fi;
  226. db_fi.set_index_id(6ul);
  227. db_fi.set_max_sequence(10l);
  228. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  229. folder->get_folder_infos().put(folder_info_peer);
  230. }
  231. auto cc = std::make_unique<proto::ClusterConfig>();
  232. auto p_folder = cc->add_folders();
  233. p_folder->set_id(std::string(folder->get_id()));
  234. p_folder->set_label(std::string(folder->get_label()));
  235. auto p_peer = p_folder->add_devices();
  236. p_peer->set_id(std::string(peer_id.get_sha256()));
  237. p_peer->set_name(std::string(peer_device->get_name()));
  238. SECTION("peer index has changed") {
  239. p_peer->set_max_sequence(123456u);
  240. p_peer->set_index_id(7ul);
  241. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  242. REQUIRE(diff_opt);
  243. auto &diff = diff_opt.value();
  244. auto r_a = diff->apply(*cluster);
  245. REQUIRE(r_a);
  246. auto fi = folder->get_folder_infos().by_device(*peer_device);
  247. REQUIRE(fi->get_index() == 7ul);
  248. REQUIRE(fi->get_max_sequence() == 123456u);
  249. bool visited = false;
  250. auto visitor = my_cluster_update_visitor_t([&](auto &diff) {
  251. visited = true;
  252. return outcome::success();
  253. });
  254. auto r_v = diff->visit(visitor, nullptr);
  255. REQUIRE(r_v);
  256. REQUIRE(visited);
  257. REQUIRE(visitor.updated_folders == 1);
  258. }
  259. }
  260. }
  261. TEST_CASE("cluster update, reset folder", "[model]") {
  262. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  263. auto my_device = device_t::create(my_id, "my-device").value();
  264. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  265. auto peer_device = device_t::create(peer_id, "peer-device").value();
  266. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  267. cluster->get_devices().put(my_device);
  268. cluster->get_devices().put(peer_device);
  269. db::Folder db_folder;
  270. db_folder.set_id("1234-5678");
  271. db_folder.set_label("my-label");
  272. db_folder.set_path("/my/path");
  273. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  274. cluster->get_folders().put(folder);
  275. db::UnknownFolder db_u_folder;
  276. db_u_folder.mutable_folder()->set_id("1111-2222");
  277. db_u_folder.mutable_folder()->set_label("unknown");
  278. auto u_folder = unknown_folder_t::create(cluster->next_uuid(), db_u_folder, peer_device->device_id()).value();
  279. auto &unknown_folders = cluster->get_unknown_folders();
  280. unknown_folders.put(u_folder);
  281. auto folder_info_my = folder_info_ptr_t();
  282. auto folder_info_peer = folder_info_ptr_t();
  283. {
  284. db::FolderInfo db_fi;
  285. db_fi.set_index_id(5ul);
  286. db_fi.set_max_sequence(10l);
  287. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  288. }
  289. {
  290. db::FolderInfo db_fi;
  291. db_fi.set_index_id(6ul);
  292. db_fi.set_max_sequence(0l);
  293. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  294. }
  295. folder->get_folder_infos().put(folder_info_my);
  296. folder->get_folder_infos().put(folder_info_peer);
  297. auto bi1 = proto::BlockInfo();
  298. bi1.set_size(5);
  299. bi1.set_hash(utils::sha256_digest("12345").value());
  300. auto b1 = block_info_t::create(bi1).assume_value();
  301. auto bi2 = proto::BlockInfo();
  302. bi2.set_size(5);
  303. bi2.set_hash(utils::sha256_digest("67890").value());
  304. auto b2 = block_info_t::create(bi2).assume_value();
  305. auto bi3 = proto::BlockInfo();
  306. bi3.set_size(5);
  307. bi3.set_hash(utils::sha256_digest("qqqqqq").value());
  308. auto b3 = block_info_t::create(bi3).assume_value();
  309. auto &blocks_map = cluster->get_blocks();
  310. blocks_map.put(b1);
  311. blocks_map.put(b2);
  312. blocks_map.put(b3);
  313. proto::FileInfo pr_fi_my;
  314. pr_fi_my.set_name("a/b.txt");
  315. pr_fi_my.set_size(5ul);
  316. pr_fi_my.set_block_size(5ul);
  317. auto fi_my = file_info_t::create(cluster->next_uuid(), pr_fi_my, folder_info_my).value();
  318. folder_info_my->add(fi_my, false);
  319. proto::FileInfo pr_fi_peer1;
  320. pr_fi_peer1.set_name("a/c.txt");
  321. pr_fi_peer1.set_size(5ul);
  322. pr_fi_peer1.set_block_size(5ul);
  323. auto fi_peer1 = file_info_t::create(cluster->next_uuid(), pr_fi_peer1, folder_info_peer).value();
  324. folder_info_peer->add(fi_peer1, false);
  325. REQUIRE(folder_info_peer->get_file_infos().size() == 1);
  326. proto::FileInfo pr_fi_peer2;
  327. pr_fi_peer2.set_name("a/d.txt");
  328. pr_fi_peer2.set_size(10ul);
  329. pr_fi_peer2.set_block_size(5ul);
  330. auto fi_peer2 = file_info_t::create(cluster->next_uuid(), pr_fi_peer2, folder_info_peer).value();
  331. folder_info_peer->add(fi_peer2, false);
  332. REQUIRE(folder_info_peer->get_file_infos().size() == 2);
  333. fi_my->assign_block(b1, 0);
  334. fi_peer1->assign_block(b2, 0);
  335. fi_peer2->assign_block(b2, 0);
  336. fi_peer2->assign_block(b3, 1);
  337. auto cc = std::make_unique<proto::ClusterConfig>();
  338. auto p_folder = cc->add_folders();
  339. p_folder->set_id(std::string(folder->get_id()));
  340. p_folder->set_label(std::string(folder->get_label()));
  341. auto p_peer = p_folder->add_devices();
  342. p_peer->set_id(std::string(peer_id.get_sha256()));
  343. p_peer->set_name(std::string(peer_device->get_name()));
  344. p_peer->set_max_sequence(123456u);
  345. p_peer->set_index_id(7ul);
  346. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  347. REQUIRE(diff_opt);
  348. auto &diff = diff_opt.value();
  349. REQUIRE(diff->apply(*cluster));
  350. auto folder_info_peer_new = folder->get_folder_infos().by_device(*peer_device);
  351. REQUIRE(folder_info_peer_new);
  352. REQUIRE(folder_info_peer_new != folder_info_peer);
  353. REQUIRE(folder_info_peer_new->get_file_infos().size() == 0);
  354. CHECK(folder_info_peer_new->get_index() == 7ul);
  355. CHECK(folder_info_peer_new->get_max_sequence() == p_peer->max_sequence());
  356. CHECK(folder_info_peer->use_count() == 1);
  357. CHECK(folder_info_peer->get_file_infos().size() == 0);
  358. CHECK(fi_peer1->use_count() == 1);
  359. CHECK(fi_peer2->use_count() == 1);
  360. CHECK(blocks_map.size() == 1);
  361. CHECK(blocks_map.get(b1->get_hash()));
  362. bool visited = false;
  363. auto visitor = my_cluster_update_visitor_t([&](auto &diff) {
  364. visited = true;
  365. return outcome::success();
  366. });
  367. auto r_v = diff->visit(visitor, nullptr);
  368. REQUIRE(r_v);
  369. REQUIRE(visited);
  370. CHECK(visitor.remove_blocks == 1);
  371. CHECK(visitor.remove_files == 1);
  372. CHECK(visitor.remove_folders == 1);
  373. CHECK(visitor.remove_unknown_folders == 1);
  374. CHECK(visitor.updated_folders == 1);
  375. }
  376. TEST_CASE("cluster update for a folder, which was not shared", "[model]") {
  377. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  378. auto my_device = device_t::create(my_id, "my-device").value();
  379. auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  380. auto peer_device = device_t::create(peer_id, "peer-device").value();
  381. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  382. cluster->get_devices().put(my_device);
  383. cluster->get_devices().put(peer_device);
  384. db::Folder db_folder;
  385. db_folder.set_id("1234-5678");
  386. db_folder.set_label("my-label");
  387. db_folder.set_path("/my/path");
  388. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  389. cluster->get_folders().put(folder);
  390. auto folder_info_my = folder_info_ptr_t();
  391. auto folder_info_peer = folder_info_ptr_t();
  392. {
  393. db::FolderInfo db_fi;
  394. db_fi.set_index_id(5ul);
  395. db_fi.set_max_sequence(10l);
  396. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  397. folder->get_folder_infos().put(folder_info_my);
  398. }
  399. auto cc = std::make_unique<proto::ClusterConfig>();
  400. auto p_folder = cc->add_folders();
  401. p_folder->set_id(std::string(folder->get_id()));
  402. p_folder->set_label(std::string(folder->get_label()));
  403. auto p_peer = p_folder->add_devices();
  404. p_peer->set_id(std::string(peer_id.get_sha256()));
  405. p_peer->set_name(std::string(peer_device->get_name()));
  406. p_peer->set_max_sequence(123456u);
  407. p_peer->set_index_id(7ul);
  408. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  409. REQUIRE(diff_opt);
  410. auto &diff = diff_opt.value();
  411. REQUIRE(diff->apply(*cluster));
  412. }
  413. TEST_CASE("cluster update with unknown devices", "[model]") {
  414. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  415. auto my_device = device_t::create(my_id, "my-device").value();
  416. auto peer_id_1 =
  417. device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  418. auto peer_id_2 =
  419. device_id_t::from_string("EAMTZPW-Q4QYERN-D57DHFS-AUP2OMG-PAHOR3R-ZWLKGAA-WQC5SVW-UJ5NXQA").value();
  420. auto peer_device = device_t::create(peer_id_1, "peer-device").value();
  421. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  422. cluster->get_devices().put(my_device);
  423. cluster->get_devices().put(peer_device);
  424. db::Folder db_folder;
  425. db_folder.set_id("1234-5678");
  426. db_folder.set_label("my-label");
  427. db_folder.set_path("/my/path");
  428. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  429. cluster->get_folders().put(folder);
  430. auto folder_info_my = folder_info_ptr_t();
  431. auto folder_info_peer = folder_info_ptr_t();
  432. {
  433. db::FolderInfo db_fi;
  434. db_fi.set_index_id(5ul);
  435. db_fi.set_max_sequence(10l);
  436. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  437. folder->get_folder_infos().put(folder_info_my);
  438. }
  439. {
  440. db::FolderInfo db_fi;
  441. db_fi.set_index_id(6ul);
  442. db_fi.set_max_sequence(0l);
  443. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  444. folder->get_folder_infos().put(folder_info_my);
  445. }
  446. folder->get_folder_infos().put(folder_info_my);
  447. folder->get_folder_infos().put(folder_info_peer);
  448. auto cc = std::make_unique<proto::ClusterConfig>();
  449. auto p_folder = cc->add_folders();
  450. p_folder->set_id(std::string(folder->get_id()));
  451. p_folder->set_label(std::string(folder->get_label()));
  452. auto p_peer_1 = p_folder->add_devices();
  453. p_peer_1->set_id(std::string(peer_id_1.get_sha256()));
  454. p_peer_1->set_name(std::string(peer_device->get_name()));
  455. p_peer_1->set_max_sequence(123456u);
  456. p_peer_1->set_index_id(7ul);
  457. auto p_peer_2 = p_folder->add_devices();
  458. p_peer_2->set_id(std::string(peer_id_2.get_sha256()));
  459. p_peer_2->set_name(std::string(peer_device->get_name()));
  460. p_peer_2->set_max_sequence(123456u);
  461. p_peer_2->set_index_id(7ul);
  462. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  463. REQUIRE(diff_opt);
  464. }
  465. TEST_CASE("cluster update nothing shared", "[model]") {
  466. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  467. auto my_device = device_t::create(my_id, "my-device").value();
  468. auto peer_id_1 =
  469. device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  470. auto peer_id_2 =
  471. device_id_t::from_string("EAMTZPW-Q4QYERN-D57DHFS-AUP2OMG-PAHOR3R-ZWLKGAA-WQC5SVW-UJ5NXQA").value();
  472. auto peer_device = device_t::create(peer_id_1, "peer-device").value();
  473. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  474. cluster->get_devices().put(my_device);
  475. cluster->get_devices().put(peer_device);
  476. auto &blocks_map = cluster->get_blocks();
  477. auto bi1 = proto::BlockInfo();
  478. bi1.set_size(5);
  479. bi1.set_hash(utils::sha256_digest("12345").value());
  480. auto b1 = block_info_t::create(bi1).assume_value();
  481. blocks_map.put(b1);
  482. db::Folder db_folder;
  483. db_folder.set_id("1234-5678");
  484. db_folder.set_label("my-label");
  485. db_folder.set_path("/my/path");
  486. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  487. cluster->get_folders().put(folder);
  488. auto folder_info_my = folder_info_ptr_t();
  489. auto folder_info_peer = folder_info_ptr_t();
  490. {
  491. db::FolderInfo db_fi;
  492. db_fi.set_index_id(5ul);
  493. db_fi.set_max_sequence(10l);
  494. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  495. folder->get_folder_infos().put(folder_info_my);
  496. }
  497. {
  498. db::FolderInfo db_fi;
  499. db_fi.set_index_id(6ul);
  500. db_fi.set_max_sequence(0l);
  501. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  502. folder->get_folder_infos().put(folder_info_peer);
  503. proto::FileInfo pr_fi_peer;
  504. pr_fi_peer.set_name("a/c.txt");
  505. pr_fi_peer.set_size(5ul);
  506. pr_fi_peer.set_block_size(5ul);
  507. auto fi_peer = file_info_t::create(cluster->next_uuid(), pr_fi_peer, folder_info_peer).value();
  508. folder_info_peer->add(fi_peer, false);
  509. fi_peer->assign_block(b1, 0);
  510. REQUIRE(folder_info_peer->get_file_infos().size() == 1);
  511. }
  512. auto cc = std::make_unique<proto::ClusterConfig>();
  513. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  514. REQUIRE(diff_opt);
  515. auto opt = diff_opt.value()->apply(*cluster);
  516. REQUIRE(opt);
  517. CHECK(blocks_map.size() == 0);
  518. CHECK(folder_info_peer->get_file_infos().size() == 0);
  519. CHECK(folder->is_shared_with(*peer_device));
  520. CHECK(folder->is_shared_with(*peer_device)->get_file_infos().size() == 0);
  521. }
  522. TEST_CASE("cluster update with remote folders", "[model]") {
  523. auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
  524. auto my_device = device_t::create(my_id, "my-device").value();
  525. auto peer_id_1 =
  526. device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
  527. auto peer_device = device_t::create(peer_id_1, "peer-device").value();
  528. auto cluster = cluster_ptr_t(new cluster_t(my_device, 1, 1));
  529. cluster->get_devices().put(my_device);
  530. cluster->get_devices().put(peer_device);
  531. db::Folder db_folder;
  532. db_folder.set_id("1234-5678");
  533. db_folder.set_label("my-label");
  534. db_folder.set_path("/my/path");
  535. auto folder = folder_t::create(cluster->next_uuid(), db_folder).value();
  536. cluster->get_folders().put(folder);
  537. auto folder_info_my = folder_info_ptr_t();
  538. auto folder_info_peer = folder_info_ptr_t();
  539. {
  540. db::FolderInfo db_fi;
  541. db_fi.set_index_id(5ul);
  542. db_fi.set_max_sequence(10l);
  543. folder_info_my = folder_info_t::create(cluster->next_uuid(), db_fi, my_device, folder).value();
  544. folder->get_folder_infos().put(folder_info_my);
  545. }
  546. {
  547. db::FolderInfo db_fi;
  548. db_fi.set_index_id(6ul);
  549. db_fi.set_max_sequence(0l);
  550. folder_info_peer = folder_info_t::create(cluster->next_uuid(), db_fi, peer_device, folder).value();
  551. folder->get_folder_infos().put(folder_info_my);
  552. }
  553. folder->get_folder_infos().put(folder_info_my);
  554. folder->get_folder_infos().put(folder_info_peer);
  555. auto cc = std::make_unique<proto::ClusterConfig>();
  556. auto p_folder = cc->add_folders();
  557. p_folder->set_id(std::string(folder->get_id()));
  558. p_folder->set_label(std::string(folder->get_label()));
  559. auto p_peer_1 = p_folder->add_devices();
  560. p_peer_1->set_id(std::string(peer_id_1.get_sha256()));
  561. p_peer_1->set_name(std::string(peer_device->get_name()));
  562. p_peer_1->set_max_sequence(123456u);
  563. p_peer_1->set_index_id(7ul);
  564. auto p_peer_my = p_folder->add_devices();
  565. p_peer_my->set_id(std::string(my_id.get_sha256()));
  566. p_peer_my->set_name(std::string(my_device->get_name()));
  567. p_peer_my->set_max_sequence(3);
  568. p_peer_my->set_index_id(5ul);
  569. auto diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  570. REQUIRE(diff_opt);
  571. auto opt = diff_opt.value()->apply(*cluster);
  572. REQUIRE(opt);
  573. auto remote_folder = peer_device->get_remote_folder_infos().by_folder(*folder);
  574. REQUIRE(remote_folder);
  575. CHECK(remote_folder->get_index() == 5ul);
  576. CHECK(remote_folder->get_max_sequence() == 3);
  577. SECTION("unshare by peer") {
  578. auto cc = std::make_unique<proto::ClusterConfig>();
  579. diff_opt = diff::peer::cluster_update_t::create(*cluster, *peer_device, *cc);
  580. REQUIRE(diff_opt);
  581. auto &diff = diff_opt.value();
  582. REQUIRE(diff->apply(*cluster));
  583. CHECK(peer_device->get_remote_folder_infos().size() == 0);
  584. auto fi = folder->get_folder_infos().by_device(*peer_device);
  585. // we are still sharing the folder with peer
  586. CHECK(fi);
  587. }
  588. }
  589. int _init() {
  590. utils::set_default("trace");
  591. return 1;
  592. }
  593. static int v = _init();