Files
syncspirit/tests/054-updates_streamer.cpp
T
2025-12-13 16:20:23 +03:00

171 lines
6.2 KiB
C++

// SPDX-License-Identifier: GPL-3.0-or-later
// SPDX-FileCopyrightText: 2023-2025 Ivan Baidakou
#include "test-utils.h"
#include "access.h"
#include "diff-builder.h"
#include "model/misc/updates_streamer.h"
using namespace syncspirit;
using namespace syncspirit::test;
using namespace syncspirit::utils;
using namespace syncspirit::model;
using update_t = model::updates_streamer_t::update_t;
TEST_CASE("updates_streamer", "[model]") {
test::init_logging();
auto my_id = device_id_t::from_string("KHQNO2S-5QSILRK-YX4JZZ4-7L77APM-QNVGZJT-EKU7IFI-PNEPBMY-4MXFMQD").value();
auto my_device = device_t::create(my_id, "my-device").value();
auto peer_id = device_id_t::from_string("VUV42CZ-IQD5A37-RPEBPM4-VVQK6E4-6WSKC7B-PVJQHHD-4PZD44V-ENC6WAZ").value();
auto peer_device = device_t::create(peer_id, "peer-device").value();
auto cluster = cluster_ptr_t(new cluster_t(my_device, 1));
auto sequencer = make_sequencer(4);
cluster->get_devices().put(my_device);
cluster->get_devices().put(peer_device);
auto builder = diff_builder_t(*cluster);
auto &folders = cluster->get_folders();
REQUIRE(builder.upsert_folder("1234-5678", "/my/path").apply());
REQUIRE(builder.share_folder(peer_id.get_sha256(), "1234-5678").apply());
auto folder = folders.by_id("1234-5678");
auto add_remote = [&](std::uint64_t index, std::int64_t sequence) {
auto &views = peer_device->get_remote_view_map();
views.push(*my_device, *folder, index, sequence);
};
SECTION("trivial") {
SECTION("no files") {
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
REQUIRE(streamer.next() == update_t({}, {}, false));
}
add_remote(0, 0);
SECTION("no files (2)") {
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
REQUIRE(streamer.next() == update_t({}, {}, false));
}
}
auto my_folder = folder->get_folder_infos().by_device(*my_device);
int seq = 1;
auto add_file = [&](const char *name) {
auto pr_file = proto::FileInfo();
proto::set_name(pr_file, name);
proto::set_sequence(pr_file, seq++);
auto &v = proto::get_version(pr_file);
proto::add_counters(v, proto::Counter(my_device->device_id().get_uint(), 0));
auto f = file_info_t::create(sequencer->next_uuid(), pr_file, my_folder).value();
my_folder->add_strict(f);
return f;
};
SECTION("2 files, index mismatch") {
SECTION("zero sequence") { add_remote(0, 0); }
SECTION("non-zero sequence") { add_remote(0, seq + 100); }
auto f1 = add_file("a.txt");
auto f2 = add_file("b.txt");
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
CHECK(streamer.next() == update_t(f1, my_folder.get(), true));
CHECK(streamer.next() == update_t(f2, my_folder.get(), false));
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("2 files, index matches, sequence greater") {
auto f1 = add_file("a.txt");
auto f2 = add_file("b.txt");
add_remote(my_folder->get_index(), f2->get_sequence());
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("2 files, index matches, sequence greater") {
auto f1 = add_file("a.txt");
auto f2 = add_file("b.txt");
add_remote(my_folder->get_index(), f1->get_sequence());
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
CHECK(streamer.next() == update_t(f2, my_folder.get(), false));
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("1 file, streamer is updated lazily") {
add_remote(my_folder->get_index(), seq - 1);
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
REQUIRE(streamer.next() == update_t({}, {}, false));
auto f1 = add_file("a.txt");
streamer.on_update(*f1, *my_folder);
CHECK(streamer.next() == update_t(f1, my_folder.get(), true));
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("empty streamer ignores updates") {
add_remote(0, seq);
auto peer_folder = folder->get_folder_infos().by_device(*peer_device);
REQUIRE(builder.unshare_folder(*peer_folder).apply());
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
REQUIRE(streamer.next() == update_t({}, {}, false));
auto f1 = add_file("a.txt");
streamer.on_update(*f1, *my_folder);
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("2 files, streamer is updated") {
add_remote(my_folder->get_index(), 0);
auto f1 = add_file("a.txt");
auto f2 = add_file("b.txt");
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
f1->set_sequence(++seq);
my_folder->add_strict(f1);
streamer.on_update(*f1, *my_folder);
CHECK(streamer.next() == update_t(f2, my_folder.get(), true));
CHECK(streamer.next() == update_t(f1, my_folder.get(), false));
REQUIRE(streamer.next() == update_t({}, {}, false));
}
SECTION("no files streaming for receive-only folder") {
SECTION("lazy streamer update") {
add_remote(0, seq);
auto &folder_type = ((model::folder_data_t *)folder.get())->access<test::to::folder_type>();
folder_type = db::FolderType::receive;
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
REQUIRE(streamer.next() == update_t({}, {}, false));
auto f1 = add_file("a.txt");
streamer.on_update(*f1, *my_folder);
CHECK(streamer.next() == update_t({}, {}, false));
}
SECTION("non-lazy streamer update") {
add_remote(0, seq);
auto f1 = add_file("a.txt");
auto &folder_type = ((model::folder_data_t *)folder.get())->access<test::to::folder_type>();
folder_type = db::FolderType::receive;
auto streamer = model::updates_streamer_t(*cluster, *peer_device);
CHECK(streamer.next() == update_t({}, {}, false));
}
}
}