diff options
Diffstat (limited to 'libs/tdlib/td/tdutils/test')
-rw-r--r-- | libs/tdlib/td/tdutils/test/Enumerator.cpp | 24 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/HazardPointers.cpp | 58 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/MpmcQueue.cpp | 205 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/MpmcWaiter.cpp | 117 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/MpscLinkQueue.cpp | 115 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/OrderedEventsProcessor.cpp | 36 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/SharedObjectPool.cpp | 96 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/crypto.cpp | 166 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/filesystem.cpp | 41 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/gzip.cpp | 113 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/heap.cpp | 178 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/json.cpp | 94 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/misc.cpp | 262 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/pq.cpp | 118 | ||||
-rw-r--r-- | libs/tdlib/td/tdutils/test/variant.cpp | 75 |
15 files changed, 1698 insertions, 0 deletions
diff --git a/libs/tdlib/td/tdutils/test/Enumerator.cpp b/libs/tdlib/td/tdutils/test/Enumerator.cpp new file mode 100644 index 0000000000..b617485462 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/Enumerator.cpp @@ -0,0 +1,24 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/Enumerator.h" +#include "td/utils/tests.h" + +TEST(Enumerator, simple) { + td::Enumerator<std::string> e; + auto b = e.add("b"); + auto a = e.add("a"); + auto d = e.add("d"); + auto c = e.add("c"); + ASSERT_STREQ(e.get(a), "a"); + ASSERT_STREQ(e.get(b), "b"); + ASSERT_STREQ(e.get(c), "c"); + ASSERT_STREQ(e.get(d), "d"); + ASSERT_EQ(a, e.add("a")); + ASSERT_EQ(b, e.add("b")); + ASSERT_EQ(c, e.add("c")); + ASSERT_EQ(d, e.add("d")); +} diff --git a/libs/tdlib/td/tdutils/test/HazardPointers.cpp b/libs/tdlib/td/tdutils/test/HazardPointers.cpp new file mode 100644 index 0000000000..36b0570530 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/HazardPointers.cpp @@ -0,0 +1,58 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/HazardPointers.h" +#include "td/utils/logging.h" +#include "td/utils/port/thread.h" +#include "td/utils/Random.h" +#include "td/utils/tests.h" + +#include <atomic> + +#if !TD_THREAD_UNSUPPORTED +TEST(HazardPointers, stress) { + struct Node { + std::atomic<std::string *> name_; + char pad[64]; + }; + int threads_n = 10; + std::vector<Node> nodes(threads_n); + td::HazardPointers<std::string> hazard_pointers(threads_n); + std::vector<td::thread> threads(threads_n); + int thread_id = 0; + for (auto &thread : threads) { + thread = td::thread([&, thread_id] { + auto holder = hazard_pointers.get_holder(thread_id, 0); + for (int i = 0; i < 1000000; i++) { + auto &node = nodes[td::Random::fast(0, threads_n - 1)]; + auto *str = holder.protect(node.name_); + if (str) { + CHECK(*str == "one" || *str == "twotwo"); + } + holder.clear(); + if (td::Random::fast(0, 5) == 0) { + std::string *new_str = new std::string(td::Random::fast(0, 1) == 0 ? "one" : "twotwo"); + if (node.name_.compare_exchange_strong(str, new_str, std::memory_order_acq_rel)) { + hazard_pointers.retire(thread_id, str); + } else { + delete new_str; + } + } + } + }); + thread_id++; + } + for (auto &thread : threads) { + thread.join(); + } + LOG(ERROR) << "Undeleted pointers: " << hazard_pointers.to_delete_size_unsafe(); + CHECK(static_cast<int>(hazard_pointers.to_delete_size_unsafe()) <= threads_n * threads_n); + for (int i = 0; i < threads_n; i++) { + hazard_pointers.retire(i); + } + CHECK(hazard_pointers.to_delete_size_unsafe() == 0); +} +#endif //!TD_THREAD_UNSUPPORTED diff --git a/libs/tdlib/td/tdutils/test/MpmcQueue.cpp b/libs/tdlib/td/tdutils/test/MpmcQueue.cpp new file mode 100644 index 0000000000..2da3f0cd3f --- /dev/null +++ b/libs/tdlib/td/tdutils/test/MpmcQueue.cpp @@ -0,0 +1,205 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/logging.h" +#include "td/utils/MpmcQueue.h" +#include "td/utils/port/thread.h" +#include "td/utils/tests.h" + +#include <algorithm> +#include <tuple> + +TEST(OneValue, simple) { + { + std::string x{"hello"}; + td::OneValue<std::string> value; + auto status = value.set_value(x); + CHECK(status); + CHECK(x.empty()); + status = value.get_value(x); + CHECK(status); + CHECK(x == "hello"); + } + { + td::OneValue<std::string> value; + std::string x; + auto status = value.get_value(x); + CHECK(!status); + CHECK(x.empty()); + std::string y{"hello"}; + status = value.set_value(y); + CHECK(!status); + CHECK(y == "hello"); + } +} + +#if !TD_THREAD_UNSUPPORTED +TEST(OneValue, stress) { + td::Stage run; + td::Stage check; + + std::string from; + bool set_status; + + std::string to; + bool get_status; + std::vector<td::thread> threads; + td::OneValue<std::string> value; + for (size_t i = 0; i < 2; i++) { + threads.push_back(td::thread([&, id = i] { + for (td::uint64 round = 1; round < 100000; round++) { + if (id == 0) { + value.reset(); + to = ""; + from = ""; + } + run.wait(round * 2); + if (id == 0) { + from = "hello"; + set_status = value.set_value(from); + } else { + get_status = value.get_value(to); + } + check.wait(round * 2); + if (id == 0) { + if (set_status) { + CHECK(get_status); + CHECK(from.empty()); + CHECK(to == "hello") << to; + } else { + CHECK(!get_status); + CHECK(from == "hello"); + CHECK(to.empty()); + } + } + } + })); + } + for (auto &thread : threads) { + thread.join(); + } +} +#endif //!TD_THREAD_UNSUPPORTED + +TEST(MpmcQueueBlock, simple) { + // Test doesn't work now and it is ok, try_pop, logic changed + return; + td::MpmcQueueBlock<std::string> block(2); + std::string x = "hello"; + using PushStatus = td::MpmcQueueBlock<std::string>::PushStatus; + using PopStatus = td::MpmcQueueBlock<std::string>::PopStatus; + auto push_status = block.push(x); + CHECK(push_status == PushStatus::Ok); + CHECK(x.empty()); + auto pop_status = block.pop(x); + CHECK(pop_status == PopStatus::Ok); + CHECK(x == "hello"); + pop_status = block.try_pop(x); + CHECK(pop_status == PopStatus::Empty); + x = "hello"; + push_status = block.push(x); + CHECK(push_status == PushStatus::Ok); + x = "hello"; + push_status = block.push(x); + CHECK(push_status == PushStatus::Closed); + CHECK(x == "hello"); + x = ""; + pop_status = block.try_pop(x); + CHECK(pop_status == PopStatus::Ok); + pop_status = block.try_pop(x); + CHECK(pop_status == PopStatus::Closed); +} + +TEST(MpmcQueue, simple) { + td::MpmcQueue<int> q(2, 1); + for (int t = 0; t < 2; t++) { + for (int i = 0; i < 100; i++) { + q.push(i, 0); + } + for (int i = 0; i < 100; i++) { + int x = q.pop(0); + CHECK(x == i) << x << " expected " << i; + } + } +} + +#if !TD_THREAD_UNSUPPORTED +TEST(MpmcQueue, multi_thread) { + size_t n = 10; + size_t m = 10; + struct Data { + size_t from{0}; + size_t value{0}; + }; + struct ThreadData { + std::vector<Data> v; + char pad[64]; + }; + td::MpmcQueue<Data> q(1024, n + m + 1); + std::vector<td::thread> n_threads(n); + std::vector<td::thread> m_threads(m); + std::vector<ThreadData> thread_data(m); + size_t thread_id = 0; + for (auto &thread : m_threads) { + thread = td::thread([&, thread_id] { + while (true) { + auto data = q.pop(thread_id); + if (data.value == 0) { + return; + } + thread_data[thread_id].v.push_back(data); + } + }); + thread_id++; + } + size_t qn = 100000; + for (auto &thread : n_threads) { + thread = td::thread([&, thread_id] { + for (size_t i = 0; i < qn; i++) { + Data data; + data.from = thread_id - m; + data.value = i + 1; + q.push(data, thread_id); + } + }); + thread_id++; + } + for (auto &thread : n_threads) { + thread.join(); + } + for (size_t i = 0; i < m; i++) { + Data data; + data.from = 0; + data.value = 0; + q.push(data, thread_id); + } + for (auto &thread : m_threads) { + thread.join(); + } + std::vector<Data> all; + for (size_t i = 0; i < m; i++) { + std::vector<size_t> from(n, 0); + for (auto &data : thread_data[i].v) { + all.push_back(data); + CHECK(data.value > from[data.from]); + from[data.from] = data.value; + } + } + CHECK(all.size() == n * qn) << all.size(); + std::sort(all.begin(), all.end(), + [](const auto &a, const auto &b) { return std::tie(a.from, a.value) < std::tie(b.from, b.value); }); + for (size_t i = 0; i < n * qn; i++) { + CHECK(all[i].from == i / qn); + CHECK(all[i].value == i % qn + 1); + } + LOG(ERROR) << "Undeleted pointers: " << q.hazard_pointers_to_delele_size_unsafe(); + CHECK(q.hazard_pointers_to_delele_size_unsafe() <= (n + m + 1) * (n + m + 1)); + for (size_t id = 0; id < n + m + 1; id++) { + q.gc(id); + } + CHECK(q.hazard_pointers_to_delele_size_unsafe() == 0) << q.hazard_pointers_to_delele_size_unsafe(); +} +#endif //!TD_THREAD_UNSUPPORTED diff --git a/libs/tdlib/td/tdutils/test/MpmcWaiter.cpp b/libs/tdlib/td/tdutils/test/MpmcWaiter.cpp new file mode 100644 index 0000000000..e27e217713 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/MpmcWaiter.cpp @@ -0,0 +1,117 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/MpmcWaiter.h" +#include "td/utils/port/sleep.h" +#include "td/utils/port/thread.h" +#include "td/utils/Random.h" +#include "td/utils/tests.h" + +#include <atomic> + +#if !TD_THREAD_UNSUPPORTED +TEST(MpmcWaiter, stress_one_one) { + td::Stage run; + td::Stage check; + + std::vector<td::thread> threads; + std::atomic<size_t> value; + size_t write_cnt = 10; + std::unique_ptr<td::MpmcWaiter> waiter; + size_t threads_n = 2; + for (size_t i = 0; i < threads_n; i++) { + threads.push_back(td::thread([&, id = static_cast<td::uint32>(i)] { + for (td::uint64 round = 1; round < 100000; round++) { + if (id == 0) { + value = 0; + waiter = std::make_unique<td::MpmcWaiter>(); + write_cnt = td::Random::fast(1, 10); + } + run.wait(round * threads_n); + if (id == 1) { + for (size_t i = 0; i < write_cnt; i++) { + value.store(i + 1, std::memory_order_relaxed); + waiter->notify(); + } + } else { + int yields = 0; + for (size_t i = 1; i <= write_cnt; i++) { + while (true) { + auto x = value.load(std::memory_order_relaxed); + if (x >= i) { + break; + } + yields = waiter->wait(yields, id); + } + yields = waiter->stop_wait(yields, id); + } + } + check.wait(round * threads_n); + } + })); + } + for (auto &thread : threads) { + thread.join(); + } +} +TEST(MpmcWaiter, stress) { + td::Stage run; + td::Stage check; + + std::vector<td::thread> threads; + size_t write_n; + size_t read_n; + std::atomic<size_t> write_pos; + std::atomic<size_t> read_pos; + size_t end_pos; + size_t write_cnt; + size_t threads_n = 20; + std::unique_ptr<td::MpmcWaiter> waiter; + for (size_t i = 0; i < threads_n; i++) { + threads.push_back(td::thread([&, id = static_cast<td::uint32>(i)] { + for (td::uint64 round = 1; round < 1000; round++) { + if (id == 0) { + write_n = td::Random::fast(1, 10); + read_n = td::Random::fast(1, 10); + write_cnt = td::Random::fast(1, 50); + end_pos = write_n * write_cnt; + write_pos = 0; + read_pos = 0; + waiter = std::make_unique<td::MpmcWaiter>(); + } + run.wait(round * threads_n); + if (id <= write_n) { + for (size_t i = 0; i < write_cnt; i++) { + if (td::Random::fast(0, 20) == 0) { + td::usleep_for(td::Random::fast(1, 300)); + } + write_pos.fetch_add(1, std::memory_order_relaxed); + waiter->notify(); + } + } else if (id > 10 && id - 10 <= read_n) { + int yields = 0; + while (true) { + auto x = read_pos.load(std::memory_order_relaxed); + if (x == end_pos) { + break; + } + if (x == write_pos.load(std::memory_order_relaxed)) { + yields = waiter->wait(yields, id); + continue; + } + yields = waiter->stop_wait(yields, id); + read_pos.compare_exchange_strong(x, x + 1, std::memory_order_relaxed); + } + } + check.wait(round * threads_n); + } + })); + } + for (auto &thread : threads) { + thread.join(); + } +} +#endif // !TD_THREAD_UNSUPPORTED diff --git a/libs/tdlib/td/tdutils/test/MpscLinkQueue.cpp b/libs/tdlib/td/tdutils/test/MpscLinkQueue.cpp new file mode 100644 index 0000000000..629e5b7223 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/MpscLinkQueue.cpp @@ -0,0 +1,115 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/format.h" +#include "td/utils/logging.h" +#include "td/utils/MpscLinkQueue.h" +#include "td/utils/port/thread.h" +#include "td/utils/tests.h" + +class NodeX : public td::MpscLinkQueueImpl::Node { + public: + explicit NodeX(int value) : value_(value) { + } + td::MpscLinkQueueImpl::Node *to_mpsc_link_queue_node() { + return static_cast<td::MpscLinkQueueImpl::Node *>(this); + } + static NodeX *from_mpsc_link_queue_node(td::MpscLinkQueueImpl::Node *node) { + return static_cast<NodeX *>(node); + } + int value() { + return value_; + } + + private: + int value_; +}; +using QueueNode = td::MpscLinkQueueUniquePtrNode<NodeX>; + +QueueNode create_node(int value) { + return QueueNode(std::make_unique<NodeX>(value)); +} + +TEST(MpscLinkQueue, one_thread) { + td::MpscLinkQueue<QueueNode> queue; + + { + queue.push(create_node(1)); + queue.push(create_node(2)); + queue.push(create_node(3)); + td::MpscLinkQueue<QueueNode>::Reader reader; + queue.pop_all(reader); + queue.push(create_node(4)); + queue.pop_all(reader); + std::vector<int> v; + while (auto node = reader.read()) { + v.push_back(node.value().value()); + } + CHECK((v == std::vector<int>{1, 2, 3, 4})) << td::format::as_array(v); + + v.clear(); + queue.push(create_node(5)); + queue.pop_all(reader); + while (auto node = reader.read()) { + v.push_back(node.value().value()); + } + CHECK((v == std::vector<int>{5})) << td::format::as_array(v); + } + + { + queue.push_unsafe(create_node(3)); + queue.push_unsafe(create_node(2)); + queue.push_unsafe(create_node(1)); + queue.push_unsafe(create_node(0)); + td::MpscLinkQueue<QueueNode>::Reader reader; + queue.pop_all_unsafe(reader); + std::vector<int> v; + while (auto node = reader.read()) { + v.push_back(node.value().value()); + } + CHECK((v == std::vector<int>{3, 2, 1, 0})) << td::format::as_array(v); + } +} + +#if !TD_THREAD_UNSUPPORTED +TEST(MpscLinkQueue, multi_thread) { + td::MpscLinkQueue<QueueNode> queue; + int threads_n = 10; + int queries_n = 1000000; + std::vector<int> next_value(threads_n); + std::vector<td::thread> threads(threads_n); + int thread_i = 0; + for (auto &thread : threads) { + thread = td::thread([&, id = thread_i] { + for (int i = 0; i < queries_n; i++) { + queue.push(create_node(i * threads_n + id)); + } + }); + thread_i++; + } + + int active_threads = threads_n; + + td::MpscLinkQueue<QueueNode>::Reader reader; + while (active_threads) { + queue.pop_all(reader); + while (auto value = reader.read()) { + auto x = value.value().value(); + auto thread_id = x % threads_n; + x /= threads_n; + CHECK(next_value[thread_id] == x); + next_value[thread_id]++; + if (x + 1 == queries_n) { + active_threads--; + } + } + } + + for (auto &thread : threads) { + thread.join(); + } +} +#endif //!TD_THREAD_UNSUPPORTED diff --git a/libs/tdlib/td/tdutils/test/OrderedEventsProcessor.cpp b/libs/tdlib/td/tdutils/test/OrderedEventsProcessor.cpp new file mode 100644 index 0000000000..6a5a20015f --- /dev/null +++ b/libs/tdlib/td/tdutils/test/OrderedEventsProcessor.cpp @@ -0,0 +1,36 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/OrderedEventsProcessor.h" +#include "td/utils/Random.h" +#include "td/utils/tests.h" + +#include <algorithm> +#include <utility> +#include <vector> + +TEST(OrderedEventsProcessor, random) { + int d = 5001; + int n = 1000000; + int offset = 1000000; + std::vector<std::pair<int, int>> v; + for (int i = 0; i < n; i++) { + auto shift = td::Random::fast(0, 1) ? td::Random::fast(0, d) : td::Random::fast(0, 1) * d; + v.push_back({i + shift, i + offset}); + } + std::sort(v.begin(), v.end()); + + td::OrderedEventsProcessor<int> processor(offset); + int next_pos = offset; + for (auto p : v) { + int seq_no = p.second; + processor.add(seq_no, seq_no, [&](auto seq_no, int x) { + ASSERT_EQ(x, next_pos); + next_pos++; + }); + } + ASSERT_EQ(next_pos, n + offset); +} diff --git a/libs/tdlib/td/tdutils/test/SharedObjectPool.cpp b/libs/tdlib/td/tdutils/test/SharedObjectPool.cpp new file mode 100644 index 0000000000..61d956f4e6 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/SharedObjectPool.cpp @@ -0,0 +1,96 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/logging.h" +#include "td/utils/SharedObjectPool.h" +#include "td/utils/tests.h" + +#include <memory> + +TEST(AtomicRefCnt, simple) { + td::detail::AtomicRefCnt cnt{0}; + cnt.inc(); + cnt.inc(); + CHECK(!cnt.dec()); + cnt.inc(); + CHECK(!cnt.dec()); + CHECK(cnt.dec()); + cnt.inc(); + CHECK(cnt.dec()); +} + +template <class T, class D> +using Ptr = td::detail::SharedPtr<T, D>; +class Deleter { + public: + template <class T> + void operator()(T *t) { + std::default_delete<T>()(t); + was_delete() = true; + } + static bool &was_delete() { + static bool flag = false; + return flag; + } +}; + +TEST(SharedPtr, simple) { + CHECK(!Deleter::was_delete()); + Ptr<std::string, Deleter> ptr = Ptr<std::string, Deleter>::create("hello"); + auto ptr2 = ptr; + CHECK(*ptr == "hello"); + CHECK(*ptr2 == "hello"); + ptr.reset(); + CHECK(*ptr2 == "hello"); + CHECK(ptr.empty()); + Ptr<std::string, Deleter> ptr3 = std::move(ptr2); + CHECK(ptr2.empty()); + CHECK(*ptr3 == "hello"); + ptr = ptr3; + CHECK(*ptr3 == "hello"); + ptr3.reset(); + CHECK(*ptr == "hello"); + ptr2 = std::move(ptr); + CHECK(ptr.empty()); + CHECK(*ptr2 == "hello"); + ptr2 = ptr2; + CHECK(*ptr2 == "hello"); + CHECK(!Deleter::was_delete()); + ptr2.reset(); + CHECK(Deleter::was_delete()); + CHECK(ptr2.empty()); +} + +TEST(SharedObjectPool, simple) { + class Node { + public: + Node() { + cnt()++; + }; + ~Node() { + cnt()--; + } + static int &cnt() { + static int cnt_ = 0; + return cnt_; + } + }; + { + td::SharedObjectPool<Node> pool; + pool.alloc(); + pool.alloc(); + pool.alloc(); + pool.alloc(); + pool.alloc(); + CHECK(Node::cnt() == 0); + CHECK(pool.total_size() == 1); + CHECK(pool.calc_free_size() == 1); + pool.alloc(), pool.alloc(), pool.alloc(); + CHECK(pool.total_size() == 3); + CHECK(pool.calc_free_size() == 3); + } + CHECK(Node::cnt() == 0); +} diff --git a/libs/tdlib/td/tdutils/test/crypto.cpp b/libs/tdlib/td/tdutils/test/crypto.cpp new file mode 100644 index 0000000000..faf4ef61a4 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/crypto.cpp @@ -0,0 +1,166 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/base64.h" +#include "td/utils/common.h" +#include "td/utils/crypto.h" +#include "td/utils/Slice.h" +#include "td/utils/tests.h" + +#include <limits> + +static td::vector<td::string> strings{"", "1", "short test string", td::string(1000000, 'a')}; + +#if TD_HAVE_OPENSSL +TEST(Crypto, AesCtrState) { + td::vector<td::uint32> answers1{0u, 1141589763u, 596296607u, 3673001485u, 2302125528u, + 330967191u, 2047392231u, 3537459563u, 307747798u, 2149598133u}; + td::vector<td::uint32> answers2{0u, 2053451992u, 1384063362u, 3266188502u, 2893295118u, + 780356167u, 1904947434u, 2043402406u, 472080809u, 1807109488u}; + + std::size_t i = 0; + for (auto length : {0, 1, 31, 32, 33, 9999, 10000, 10001, 999999, 1000001}) { + td::uint32 seed = length; + td::string s(length, '\0'); + for (auto &c : s) { + seed = seed * 123457567u + 987651241u; + c = static_cast<char>((seed >> 23) & 255); + } + + td::UInt256 key; + for (auto &c : key.raw) { + seed = seed * 123457567u + 987651241u; + c = (seed >> 23) & 255; + } + td::UInt128 iv; + for (auto &c : iv.raw) { + seed = seed * 123457567u + 987651241u; + c = (seed >> 23) & 255; + } + + td::AesCtrState state; + state.init(key, iv); + td::string t(length, '\0'); + state.encrypt(s, t); + ASSERT_EQ(answers1[i], td::crc32(t)); + state.init(key, iv); + state.decrypt(t, t); + ASSERT_STREQ(s, t); + + for (auto &c : iv.raw) { + c = 0xFF; + } + state.init(key, iv); + state.encrypt(s, t); + ASSERT_EQ(answers2[i], td::crc32(t)); + + i++; + } +} + +TEST(Crypto, Sha256State) { + for (auto length : {0, 1, 31, 32, 33, 9999, 10000, 10001, 999999, 1000001}) { + auto s = td::rand_string(std::numeric_limits<char>::min(), std::numeric_limits<char>::max(), length); + td::UInt256 baseline; + td::sha256(s, td::MutableSlice(baseline.raw, 32)); + + td::Sha256State state; + td::sha256_init(&state); + auto v = td::rand_split(s); + for (auto &x : v) { + td::sha256_update(x, &state); + } + td::UInt256 result; + td::sha256_final(&state, td::MutableSlice(result.raw, 32)); + ASSERT_TRUE(baseline == result); + } +} + +TEST(Crypto, PBKDF) { + td::vector<td::string> passwords{"", "qwerty", std::string(1000, 'a')}; + td::vector<td::string> salts{"", "qwerty", std::string(1000, 'a')}; + td::vector<int> iteration_counts{1, 2, 1000}; + td::vector<td::Slice> answers{ + "984LZT0tcqQQjPWr6RL/3Xd2Ftu7J6cOggTzri0Pb60=", "lzmEEdaupDp3rO+SImq4J41NsGaL0denanJfdoCsRcU=", + "T8WKIcEAzhg1uPmZHXOLVpZdFLJOF2H73/xprF4LZno=", "NHxAnMhPOATsb1wV0cGDlAIs+ofzI6I4I8eGJeWN9Qw=", + "fjYi7waEPjbVYEuZ61/Nm2hbk/vRdShoJoXg4Ygnqe4=", "GhW6e95hGJSf+ID5IrSbvzWyBZ1l35A+UoL55Uh/njk=", + "BueLDpqSCEc0GWk83WgMwz3UsWwfvVKcvllETSB/Yq8=", "hgHgJZNWRh78PyPdVJsK8whgHOHQbNQiyaTuGDX2IFo=", + "T2xdyNT1GlcA4+MVNzOe7NCgSAAzNkanNsmuoSr+4xQ=", "/f6t++GUPE+e63+0TrlInL+UsmzRSAAFopa8BBBmb2w=", + "8Zn98QEAKS9wPOUlN09+pfm0SWs1IGeQxQkNMT/1k48=", "sURLQ/6UX/KVYedyQB21oAtMJ+STZ4iwpxfQtqmWkLw=", + "T9t/EJXFpPs2Lhca7IVGphTC/OdEloPMHw1UhDnXcyQ=", "TIrtN05E9KQL6Lp/wjtbsFS+KkWZ8jlGK0ErtaoitOg=", + "+1KcMBjyUNz5VMaIfE5wkGwS6I+IQ5FhK+Ou2HgtVoQ=", "h36ci1T0vGllCl/xJxq6vI7n28Bg40dilzWOKg6Jt8k=", + "9uwsHJsotTiTqqCYftN729Dg7QI2BijIjV2MvSEUAeE=", "/l+vd/XYgbioh1SfLMaGRr13udmY6TLSlG4OYmytwGU=", + "7qfZZBbMRLtgjqq7GHgWa/UfXPajW8NXpJ6/T3P1rxI=", "ufwz94p28WnoOFdbrb1oyQEzm/v0CV2b0xBVxeEPJGA=", + "T/PUUBX2vGMUsI6httlhbMHlGPMvqFBNzayU5voVlaw=", "viMvsvTg9GfQymF3AXZ8uFYTDa3qLrqJJk9w/74iZfg=", + "HQF+rOZMW4DAdgZz8kAMe28eyIi0rs3a3u/mUeGPNfs=", "7lBVA+GnSxWF/eOo+tyyTB7niMDl1MqP8yzo+xnHTyw=", + "aTWb7HQAxaTKhSiRPY3GuM1GVmq/FPuwWBU/TUpdy70=", "fbg8M/+Ht/oU+UAZ4dQcGPo+wgCCHaA+GM4tm5jnWcY=", + "DJbCGFMIR/5neAlpda8Td5zftK4NGekVrg2xjrKW/4c="}; + + std::size_t pos = 0; + for (auto &password : passwords) { + for (auto &salt : salts) { + for (auto &iteration_count : iteration_counts) { + char result[32]; + td::pbkdf2_sha256(password, salt, iteration_count, {result, 32}); + ASSERT_STREQ(answers[pos], td::base64_encode({result, 32})); + pos++; + } + } + } +} + +TEST(Crypto, sha1) { + td::vector<td::Slice> answers{"2jmj7l5rSw0yVb/vlWAYkK/YBwk=", "NWoZK3kTsExUV00Ywo1G5jlUKKs=", + "uRysQwoax0pNJeBC3+zpQzJy1rA=", "NKqXPNTE2qT2Husr260nMWU0AW8="}; + + for (std::size_t i = 0; i < strings.size(); i++) { + unsigned char output[20]; + td::sha1(strings[i], output); + ASSERT_STREQ(answers[i], td::base64_encode(td::Slice(output, 20))); + } +} + +TEST(Crypto, sha256) { + td::vector<td::Slice> answers{ + "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", "a4ayc/80/OGda4BO/1o/V0etpOqiLx1JwB5S3beHW0s=", + "yPMaY7Q8PKPwCsw64UnDD5mhRcituEJgzLZMvr0O8pY=", "zcduXJkU+5KBocfihNc+Z/GAmkiklyAOBG05zMcRLNA="}; + + for (std::size_t i = 0; i < strings.size(); i++) { + td::string output(32, '\0'); + td::sha256(strings[i], output); + ASSERT_STREQ(answers[i], td::base64_encode(output)); + } +} + +TEST(Crypto, md5) { + td::vector<td::Slice> answers{ + "1B2M2Y8AsgTpgAmY7PhCfg==", "xMpCOKC5I4INzFCab3WEmw==", "vwBninYbDRkgk+uA7GMiIQ==", "dwfWrk4CfHDuoqk1wilvIQ=="}; + + for (std::size_t i = 0; i < strings.size(); i++) { + td::string output(16, '\0'); + td::md5(strings[i], output); + ASSERT_STREQ(answers[i], td::base64_encode(output)); + } +} +#endif + +#if TD_HAVE_ZLIB +TEST(Crypto, crc32) { + td::vector<td::uint32> answers{0u, 2212294583u, 3013144151u, 3693461436u}; + + for (std::size_t i = 0; i < strings.size(); i++) { + ASSERT_EQ(answers[i], td::crc32(strings[i])); + } +} +#endif + +TEST(Crypto, crc64) { + td::vector<td::uint64> answers{0ull, 3039664240384658157ull, 17549519902062861804ull, 8794730974279819706ull}; + + for (std::size_t i = 0; i < strings.size(); i++) { + ASSERT_EQ(answers[i], td::crc64(strings[i])); + } +} diff --git a/libs/tdlib/td/tdutils/test/filesystem.cpp b/libs/tdlib/td/tdutils/test/filesystem.cpp new file mode 100644 index 0000000000..a0a92c14eb --- /dev/null +++ b/libs/tdlib/td/tdutils/test/filesystem.cpp @@ -0,0 +1,41 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/filesystem.h" +#include "td/utils/tests.h" + +TEST(Misc, clean_filename) { + using td::clean_filename; + ASSERT_STREQ(clean_filename("-1234567"), "-1234567"); + ASSERT_STREQ(clean_filename(".git"), "git"); + ASSERT_STREQ(clean_filename("../../.git"), "git"); + ASSERT_STREQ(clean_filename(".././.."), ""); + ASSERT_STREQ(clean_filename("../"), ""); + ASSERT_STREQ(clean_filename(".."), ""); + ASSERT_STREQ(clean_filename("test/git/ as dsa . a"), "as dsa.a"); + ASSERT_STREQ(clean_filename(" . "), ""); + ASSERT_STREQ(clean_filename("!@#$%^&*()_+-=[]{;|:\"}'<>?,.`~"), "!@#$%^ ()_+-=[]{; } ,.~"); + ASSERT_STREQ(clean_filename("!@#$%^&*()_+-=[]{}\\|:\";'<>?,.`~"), "; ,.~"); + ASSERT_STREQ(clean_filename("عرفها بعد قد. هذا مع تاريخ اليميني واندونيسيا،, لعدم تاريخ لهيمنة الى"), + "عرفها بعد قد.هذا مع تاريخ اليميني"); + ASSERT_STREQ( + clean_filename( + "012345678901234567890123456789012345678901234567890123456789adsasdasdsaa.01234567890123456789asdasdasdasd"), + "012345678901234567890123456789012345678901234567890123456789.01234567890123456789"); + ASSERT_STREQ(clean_filename("01234567890123456789012345678901234567890123456789<>*?: <>*?:0123456789adsasdasdsaa. " + "0123456789`<><<>><><>0123456789asdasdasdasd"), + "01234567890123456789012345678901234567890123456789.0123456789"); + ASSERT_STREQ(clean_filename("01234567890123456789012345678901234567890123456789<>*?: <>*?:0123456789adsasdasdsaa. " + "0123456789`<><><>0123456789asdasdasdasd"), + "01234567890123456789012345678901234567890123456789.0123456789 012"); + ASSERT_STREQ(clean_filename("C:/document.tar.gz"), "document.tar.gz"); + ASSERT_STREQ(clean_filename("test...."), "test"); + ASSERT_STREQ(clean_filename("....test"), "test"); + ASSERT_STREQ(clean_filename("test.exe...."), "test.exe"); // extension has changed + ASSERT_STREQ(clean_filename("test.exe01234567890123456789...."), + "test.exe01234567890123456789"); // extension may be more then 20 characters + ASSERT_STREQ(clean_filename("....test....asdf"), "test.asdf"); +} diff --git a/libs/tdlib/td/tdutils/test/gzip.cpp b/libs/tdlib/td/tdutils/test/gzip.cpp new file mode 100644 index 0000000000..e4bd81eb0d --- /dev/null +++ b/libs/tdlib/td/tdutils/test/gzip.cpp @@ -0,0 +1,113 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/buffer.h" +#include "td/utils/ByteFlow.h" +#include "td/utils/Gzip.h" +#include "td/utils/GzipByteFlow.h" +#include "td/utils/logging.h" +#include "td/utils/Status.h" +#include "td/utils/tests.h" + +static void encode_decode(td::string s) { + auto r = td::gzencode(s, 2); + ASSERT_TRUE(!r.empty()); + if (r.empty()) { + return; + } + auto new_s = td::gzdecode(r.as_slice()); + ASSERT_TRUE(!new_s.empty()); + if (new_s.empty()) { + return; + } + ASSERT_EQ(s, new_s.as_slice().str()); +} + +TEST(Gzip, gzencode_gzdecode) { + auto str = td::rand_string(0, 127, 1000); + encode_decode(str); + str = td::rand_string('a', 'z', 1000000); + encode_decode(str); + str = td::string(1000000, 'a'); + encode_decode(str); +} + +TEST(Gzip, flow) { + auto str = td::rand_string('a', 'z', 1000000); + auto parts = td::rand_split(str); + + auto input_writer = td::ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + td::ByteFlowSource source(&input); + td::GzipByteFlow gzip_flow(td::Gzip::Encode); + gzip_flow = td::GzipByteFlow(td::Gzip::Encode); + td::ByteFlowSink sink; + + source >> gzip_flow >> sink; + + ASSERT_TRUE(!sink.is_ready()); + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + ASSERT_TRUE(!sink.is_ready()); + source.close_input(td::Status::OK()); + ASSERT_TRUE(sink.is_ready()); + ASSERT_TRUE(sink.status().is_ok()); + auto res = sink.result()->move_as_buffer_slice().as_slice().str(); + ASSERT_TRUE(!res.empty()); + ASSERT_EQ(td::gzencode(str, 2).as_slice().str(), res); +} +TEST(Gzip, flow_error) { + auto str = td::rand_string('a', 'z', 1000000); + auto zip = td::gzencode(str).as_slice().str(); + zip.resize(zip.size() - 1); + auto parts = td::rand_split(zip); + + auto input_writer = td::ChainBufferWriter(); + auto input = input_writer.extract_reader(); + td::ByteFlowSource source(&input); + td::GzipByteFlow gzip_flow(td::Gzip::Decode); + td::ByteFlowSink sink; + + source >> gzip_flow >> sink; + + ASSERT_TRUE(!sink.is_ready()); + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + ASSERT_TRUE(!sink.is_ready()); + source.close_input(td::Status::OK()); + ASSERT_TRUE(sink.is_ready()); + ASSERT_TRUE(!sink.status().is_ok()); +} + +TEST(Gzip, encode_decode_flow) { + auto str = td::rand_string('a', 'z', 1000000); + auto parts = td::rand_split(str); + auto input_writer = td::ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + td::ByteFlowSource source(&input); + td::GzipByteFlow gzip_encode_flow(td::Gzip::Encode); + td::GzipByteFlow gzip_decode_flow(td::Gzip::Decode); + td::GzipByteFlow gzip_encode_flow2(td::Gzip::Encode); + td::GzipByteFlow gzip_decode_flow2(td::Gzip::Decode); + td::ByteFlowSink sink; + source >> gzip_encode_flow >> gzip_decode_flow >> gzip_encode_flow2 >> gzip_decode_flow2 >> sink; + + ASSERT_TRUE(!sink.is_ready()); + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + ASSERT_TRUE(!sink.is_ready()); + source.close_input(td::Status::OK()); + ASSERT_TRUE(sink.is_ready()); + LOG_IF(ERROR, sink.status().is_error()) << sink.status(); + ASSERT_TRUE(sink.status().is_ok()); + ASSERT_EQ(str, sink.result()->move_as_buffer_slice().as_slice().str()); +} diff --git a/libs/tdlib/td/tdutils/test/heap.cpp b/libs/tdlib/td/tdutils/test/heap.cpp new file mode 100644 index 0000000000..0dcfcf98ff --- /dev/null +++ b/libs/tdlib/td/tdutils/test/heap.cpp @@ -0,0 +1,178 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/tests.h" + +#include "td/utils/common.h" +#include "td/utils/Heap.h" +#include "td/utils/logging.h" +#include "td/utils/Random.h" + +#include <algorithm> +#include <cstdio> +#include <cstdlib> +#include <set> +#include <utility> + +REGISTER_TESTS(heap) + +using namespace td; + +TEST(Heap, sort_random_perm) { + int n = 1000000; + std::vector<int> v(n); + for (int i = 0; i < n; i++) { + v[i] = i; + } + std::srand(123); + std::random_shuffle(v.begin(), v.end()); + std::vector<HeapNode> nodes(n); + KHeap<int> kheap; + for (int i = 0; i < n; i++) { + kheap.insert(v[i], &nodes[i]); + } + for (int i = 0; i < n; i++) { + ASSERT_EQ(i, kheap.top_key()); + kheap.pop(); + } +}; + +class CheckedHeap { + public: + void set_max_size(int max_size) { + nodes.resize(max_size); + free_ids.resize(max_size); + rev_ids.resize(max_size); + for (int i = 0; i < max_size; i++) { + free_ids[i] = max_size - i - 1; + nodes[i].value = i; + } + } + static void xx(int key, const HeapNode *heap_node) { + const Node *node = static_cast<const Node *>(heap_node); + std::fprintf(stderr, "(%d;%d)", node->key, node->value); + } + void check() const { + for (auto p : set_heap) { + std::fprintf(stderr, "(%d;%d)", p.first, p.second); + } + std::fprintf(stderr, "\n"); + kheap.for_each(xx); + std::fprintf(stderr, "\n"); + kheap.check(); + } + int random_id() const { + CHECK(!empty()); + return ids[Random::fast(0, static_cast<int>(ids.size() - 1))]; + } + size_t size() const { + return ids.size(); + } + bool empty() const { + return ids.empty(); + } + + int top_key() const { + CHECK(!empty()); + int res = set_heap.begin()->first; + ASSERT_EQ(set_heap.size(), kheap.size()); + ASSERT_EQ(res, kheap.top_key()); + return res; + } + int insert(int key) { + // std::fprintf(stderr, "insert %d\n", key); + int id; + if (free_ids.empty()) { + UNREACHABLE(); + id = static_cast<int>(nodes.size()); + nodes.emplace_back(key, id); + rev_ids.push_back(-1); + } else { + id = free_ids.back(); + free_ids.pop_back(); + nodes[id].key = key; + } + rev_ids[id] = static_cast<int>(ids.size()); + ids.push_back(id); + kheap.insert(key, &nodes[id]); + set_heap.emplace(key, id); + return id; + } + void fix_key(int new_key, int id) { + // std::fprintf(stderr, "fix key %d %d (old_key = %d)\n", new_key, id, nodes[id].key); + set_heap.erase(std::make_pair(nodes[id].key, id)); + nodes[id].key = new_key; + kheap.fix(new_key, &nodes[id]); + set_heap.emplace(new_key, id); + } + void erase(int id) { + // std::fprintf(stderr, "erase %d\n", id); + int pos = rev_ids[id]; + CHECK(pos != -1); + ids[pos] = ids.back(); + rev_ids[ids[pos]] = pos; + ids.pop_back(); + rev_ids[id] = -1; + free_ids.push_back(id); + + kheap.erase(&nodes[id]); + set_heap.erase(std::make_pair(nodes[id].key, id)); + } + void pop() { + // std::fprintf(stderr, "pop\n"); + CHECK(!empty()); + Node *node = static_cast<Node *>(kheap.pop()); + int id = node->value; + ASSERT_EQ(node->key, set_heap.begin()->first); + + int pos = rev_ids[id]; + CHECK(pos != -1); + ids[pos] = ids.back(); + rev_ids[ids[pos]] = pos; + ids.pop_back(); + rev_ids[id] = -1; + free_ids.push_back(id); + + set_heap.erase(std::make_pair(nodes[id].key, id)); + } + + private: + struct Node : public HeapNode { + Node() = default; + Node(int key, int value) : key(key), value(value) { + } + int key = 0; + int value = 0; + }; + vector<int> ids; + vector<int> rev_ids; + vector<int> free_ids; + vector<Node> nodes; + std::set<std::pair<int, int>> set_heap; + KHeap<int> kheap; +}; + +TEST(Heap, random_events) { + CheckedHeap heap; + heap.set_max_size(1000); + for (int i = 0; i < 300000; i++) { + if (!heap.empty()) { + heap.top_key(); + } + + int x = Random::fast(0, 4); + if (heap.empty() || (x < 2 && heap.size() < 1000)) { + heap.insert(Random::fast(0, 99)); + } else if (x < 3) { + heap.fix_key(Random::fast(0, 99), heap.random_id()); + } else if (x < 4) { + heap.erase(heap.random_id()); + } else if (x < 5) { + heap.pop(); + } + // heap.check(); + } +} diff --git a/libs/tdlib/td/tdutils/test/json.cpp b/libs/tdlib/td/tdutils/test/json.cpp new file mode 100644 index 0000000000..deca81a791 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/json.cpp @@ -0,0 +1,94 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/tests.h" + +#include "td/utils/JsonBuilder.h" +#include "td/utils/logging.h" +#include "td/utils/StringBuilder.h" + +#include <tuple> +#include <utility> + +REGISTER_TESTS(json) + +using namespace td; + +static void decode_encode(string str, string result = "") { + auto str_copy = str; + auto r_value = json_decode(str_copy); + ASSERT_TRUE(r_value.is_ok()); + if (r_value.is_error()) { + LOG(INFO) << r_value.error(); + return; + } + auto new_str = json_encode<string>(r_value.ok()); + if (result.empty()) { + result = str; + } + ASSERT_EQ(result, new_str); +} + +TEST(JSON, array) { + char tmp[1000]; + StringBuilder sb({tmp, sizeof(tmp)}); + JsonBuilder jb(std::move(sb)); + jb.enter_value().enter_array() << "Hello" << -123; + ASSERT_EQ(jb.string_builder().is_error(), false); + auto encoded = jb.string_builder().as_cslice().str(); + ASSERT_EQ("[\"Hello\",-123]", encoded); + decode_encode(encoded); +} +TEST(JSON, object) { + char tmp[1000]; + StringBuilder sb({tmp, sizeof(tmp)}); + JsonBuilder jb(std::move(sb)); + auto c = jb.enter_object(); + c << std::tie("key", "value"); + c << std::make_pair("1", 2); + c.leave(); + ASSERT_EQ(jb.string_builder().is_error(), false); + auto encoded = jb.string_builder().as_cslice().str(); + ASSERT_EQ("{\"key\":\"value\",\"1\":2}", encoded); + decode_encode(encoded); +} + +TEST(JSON, nested) { + char tmp[1000]; + StringBuilder sb({tmp, sizeof(tmp)}); + JsonBuilder jb(std::move(sb)); + { + auto a = jb.enter_array(); + a << 1; + { a.enter_value().enter_array() << 2; } + a << 3; + } + ASSERT_EQ(jb.string_builder().is_error(), false); + auto encoded = jb.string_builder().as_cslice().str(); + ASSERT_EQ("[1,[2],3]", encoded); + decode_encode(encoded); +} + +TEST(JSON, kphp) { + decode_encode("[]"); + decode_encode("[[]]"); + decode_encode("{}"); + decode_encode("{}"); + decode_encode("\"\\n\""); + decode_encode( + "\"" + "some long string \\t \\r \\\\ \\n \\f \\\" " + "\\u1234" + "\""); + decode_encode( + "{\"keyboard\":[[\"\\u2022 abcdefg\"],[\"\\u2022 hijklmnop\"],[\"\\u2022 " + "qrstuvwxyz\"]],\"one_time_keyboard\":true}"); + decode_encode( + " \n { \"keyboard\" : \n [[ \"\\u2022 abcdefg\" ] , \n [ \"\\u2022 hijklmnop\" \n ],[ \n \"\\u2022 " + "qrstuvwxyz\"]], \n \"one_time_keyboard\"\n:\ntrue\n}\n \n", + "{\"keyboard\":[[\"\\u2022 abcdefg\"],[\"\\u2022 hijklmnop\"],[\"\\u2022 " + "qrstuvwxyz\"]],\"one_time_keyboard\":true}"); +} diff --git a/libs/tdlib/td/tdutils/test/misc.cpp b/libs/tdlib/td/tdutils/test/misc.cpp new file mode 100644 index 0000000000..dd1f1ec457 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/misc.cpp @@ -0,0 +1,262 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/base64.h" +#include "td/utils/HttpUrl.h" +#include "td/utils/logging.h" +#include "td/utils/misc.h" +#include "td/utils/port/EventFd.h" +#include "td/utils/port/FileFd.h" +#include "td/utils/port/path.h" +#include "td/utils/port/sleep.h" +#include "td/utils/port/Stat.h" +#include "td/utils/port/thread.h" +#include "td/utils/Random.h" +#include "td/utils/Slice.h" +#include "td/utils/StringBuilder.h" +#include "td/utils/tests.h" + +#include <atomic> +#include <clocale> +#include <limits> +#include <locale> + +using namespace td; + +#if TD_LINUX || TD_DARWIN +TEST(Misc, update_atime_saves_mtime) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(ERROR)); + std::string name = "test_file"; + unlink(name).ignore(); + auto r_file = FileFd::open(name, FileFd::Read | FileFd::Flags::Create | FileFd::Flags::Truncate); + LOG_IF(ERROR, r_file.is_error()) << r_file.error(); + ASSERT_TRUE(r_file.is_ok()); + r_file.move_as_ok().close(); + + auto info = stat(name).ok(); + int32 tests_ok = 0; + int32 tests_wa = 0; + for (int i = 0; i < 10000; i++) { + update_atime(name).ensure(); + auto new_info = stat(name).ok(); + if (info.mtime_nsec_ == new_info.mtime_nsec_) { + tests_ok++; + } else { + tests_wa++; + info.mtime_nsec_ = new_info.mtime_nsec_; + } + ASSERT_EQ(info.mtime_nsec_, new_info.mtime_nsec_); + usleep_for(Random::fast(0, 1000)); + } + if (tests_wa > 0) { + LOG(ERROR) << "Access time was unexpectedly updated " << tests_wa << " times"; + } + unlink(name).ensure(); +} + +TEST(Misc, update_atime_change_atime) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(ERROR)); + std::string name = "test_file"; + unlink(name).ignore(); + auto r_file = FileFd::open(name, FileFd::Read | FileFd::Flags::Create | FileFd::Flags::Truncate); + LOG_IF(ERROR, r_file.is_error()) << r_file.error(); + ASSERT_TRUE(r_file.is_ok()); + r_file.move_as_ok().close(); + auto info = stat(name).ok(); + // not enough for fat and e.t.c. + usleep_for(5000000); + update_atime(name).ensure(); + auto new_info = stat(name).ok(); + if (info.atime_nsec_ == new_info.atime_nsec_) { + LOG(ERROR) << "Access time was unexpectedly not changed"; + } + unlink(name).ensure(); +} +#endif + +TEST(Misc, errno_tls_bug) { + // That's a problem that should be avoided + // errno = 0; + // impl_.alloc(123); + // CHECK(errno == 0); + +#if !TD_THREAD_UNSUPPORTED && !TD_EVENTFD_UNSUPPORTED + EventFd test_event_fd; + test_event_fd.init(); + std::atomic<int> s(0); + s = 1; + td::thread th([&] { + while (s != 1) { + } + test_event_fd.acquire(); + }); + th.join(); + + for (int i = 0; i < 1000; i++) { + vector<EventFd> events(10); + vector<td::thread> threads; + for (auto &event : events) { + event.init(); + event.release(); + } + for (auto &event : events) { + threads.push_back(td::thread([&] { + { + EventFd tmp; + tmp.init(); + tmp.acquire(); + } + event.acquire(); + })); + } + for (auto &thread : threads) { + thread.join(); + } + } +#endif +} + +TEST(Misc, base64) { + ASSERT_TRUE(is_base64("dGVzdA==") == true); + ASSERT_TRUE(is_base64("dGVzdB==") == false); + ASSERT_TRUE(is_base64("dGVzdA=") == false); + ASSERT_TRUE(is_base64("dGVzdA") == false); + ASSERT_TRUE(is_base64("dGVz") == true); + ASSERT_TRUE(is_base64("") == true); + ASSERT_TRUE(is_base64("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/") == true); + ASSERT_TRUE(is_base64("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=") == false); + ASSERT_TRUE(is_base64("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-/") == false); + ASSERT_TRUE(is_base64("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_") == false); + ASSERT_TRUE(is_base64("====") == false); + + ASSERT_TRUE(is_base64url("dGVzdA==") == true); + ASSERT_TRUE(is_base64url("dGVzdB==") == false); + ASSERT_TRUE(is_base64url("dGVzdA=") == false); + ASSERT_TRUE(is_base64url("dGVzdA") == true); + ASSERT_TRUE(is_base64url("dGVz") == true); + ASSERT_TRUE(is_base64url("") == true); + ASSERT_TRUE(is_base64url("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_") == true); + ASSERT_TRUE(is_base64url("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_=") == false); + ASSERT_TRUE(is_base64url("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-/") == false); + ASSERT_TRUE(is_base64url("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/") == false); + ASSERT_TRUE(is_base64url("====") == false); + + for (int l = 0; l < 300000; l += l / 20 + l / 1000 * 500 + 1) { + for (int t = 0; t < 10; t++) { + string s = rand_string(std::numeric_limits<char>::min(), std::numeric_limits<char>::max(), l); + string encoded = base64url_encode(s); + auto decoded = base64url_decode(encoded); + ASSERT_TRUE(decoded.is_ok()); + ASSERT_TRUE(decoded.ok() == s); + + encoded = base64_encode(s); + decoded = base64_decode(encoded); + ASSERT_TRUE(decoded.is_ok()); + ASSERT_TRUE(decoded.ok() == s); + } + } + + ASSERT_TRUE(base64url_decode("dGVzdA").is_ok()); + ASSERT_TRUE(base64url_decode("dGVzdB").is_error()); + ASSERT_TRUE(base64_encode(base64url_decode("dGVzdA").ok()) == "dGVzdA=="); + ASSERT_TRUE(base64_encode("any carnal pleas") == "YW55IGNhcm5hbCBwbGVhcw=="); + ASSERT_TRUE(base64_encode("any carnal pleasu") == "YW55IGNhcm5hbCBwbGVhc3U="); + ASSERT_TRUE(base64_encode("any carnal pleasur") == "YW55IGNhcm5hbCBwbGVhc3Vy"); + ASSERT_TRUE(base64_encode(" /'.;.';≤.];,].',[.;/,.;/]/..;!@#!*(%?::;!%\";") == + "ICAgICAgLycuOy4nO+KJpC5dOyxdLicsWy47LywuOy9dLy4uOyFAIyEqKCU/" + "Ojo7ISUiOw=="); +} + +TEST(Misc, to_integer) { + ASSERT_EQ(to_integer<int32>("-1234567"), -1234567); + ASSERT_EQ(to_integer<int64>("-1234567"), -1234567); + ASSERT_EQ(to_integer<uint32>("-1234567"), 0u); + ASSERT_EQ(to_integer<int16>("-1234567"), 10617); + ASSERT_EQ(to_integer<uint16>("-1234567"), 0u); + ASSERT_EQ(to_integer<int16>("-1254567"), -9383); + ASSERT_EQ(to_integer<uint16>("1254567"), 9383u); + ASSERT_EQ(to_integer<int64>("-12345678910111213"), -12345678910111213); + ASSERT_EQ(to_integer<uint64>("12345678910111213"), 12345678910111213ull); + + ASSERT_EQ(to_integer_safe<int32>("-1234567").ok(), -1234567); + ASSERT_EQ(to_integer_safe<int64>("-1234567").ok(), -1234567); + ASSERT_TRUE(to_integer_safe<uint32>("-1234567").is_error()); + ASSERT_TRUE(to_integer_safe<int16>("-1234567").is_error()); + ASSERT_TRUE(to_integer_safe<uint16>("-1234567").is_error()); + ASSERT_TRUE(to_integer_safe<int16>("-1254567").is_error()); + ASSERT_TRUE(to_integer_safe<uint16>("1254567").is_error()); + ASSERT_EQ(to_integer_safe<int64>("-12345678910111213").ok(), -12345678910111213); + ASSERT_EQ(to_integer_safe<uint64>("12345678910111213").ok(), 12345678910111213ull); + ASSERT_TRUE(to_integer_safe<uint64>("-12345678910111213").is_error()); +} + +static void test_to_double_one(CSlice str, Slice expected, int precision = 6) { + auto result = PSTRING() << td::StringBuilder::FixedDouble(to_double(str), precision); + if (expected != result) { + LOG(ERROR) << "To double conversion failed: have " << str << ", expected " << expected << ", parsed " + << to_double(str) << ", got " << result; + } +} + +static void test_to_double() { + test_to_double_one("0", "0.000000"); + test_to_double_one("1", "1.000000"); + test_to_double_one("-10", "-10.000000"); + test_to_double_one("1.234", "1.234000"); + test_to_double_one("-1.234e2", "-123.400000"); + test_to_double_one("inf", "inf"); + test_to_double_one(" inF asdasd", "inf"); + test_to_double_one(" inFasdasd", "0.000000"); + test_to_double_one(" NaN", "nan"); + test_to_double_one(" 12345678910111213141516171819 asdasd", "12345678910111213670658736128.000000"); + test_to_double_one("1.234567891011121314E123", + "1234567891011121363209105003376291141757777526749278953577304234065881343284952489418916814035346" + "625663604561924259911303168.000000"); + test_to_double_one("1.234567891011121314E-9", "0.000000"); + test_to_double_one("123456789", "123456789.000000"); + test_to_double_one("-1,234567891011121314E123", "-1.000000"); + test_to_double_one("123456789", "123456789", 0); + test_to_double_one("1.23456789", "1", 0); + test_to_double_one("1.23456789", "1.2", 1); + test_to_double_one("1.23456789", "1.23", 2); + test_to_double_one("1.23456789", "1.235", 3); + test_to_double_one("1.23456789", "1.2346", 4); + test_to_double_one("1.23456789", "1.23457", 5); + test_to_double_one("1.23456789", "1.234568", 6); + test_to_double_one("1.23456789", "1.2345679", 7); + test_to_double_one("1.23456789", "1.23456789", 8); + test_to_double_one("1.23456789", "1.234567890", 9); + test_to_double_one("1.23456789", "1.2345678900", 10); +} + +TEST(Misc, to_double) { + test_to_double(); + const char *locale_name = (std::setlocale(LC_ALL, "fr-FR") == nullptr ? "" : "fr-FR"); + std::locale new_locale(locale_name); + std::locale::global(new_locale); + test_to_double(); + std::locale::global(std::locale::classic()); + test_to_double(); +} + +static void test_get_url_query_file_name_one(const char *prefix, const char *suffix, const char *file_name) { + auto path = string(prefix) + string(file_name) + string(suffix); + ASSERT_STREQ(file_name, get_url_query_file_name(path)); + ASSERT_STREQ(file_name, get_url_file_name("http://telegram.org" + path)); + ASSERT_STREQ(file_name, get_url_file_name("http://telegram.org:80" + path)); + ASSERT_STREQ(file_name, get_url_file_name("telegram.org" + path)); +} + +TEST(Misc, get_url_query_file_name) { + for (auto suffix : {"?t=1#test", "#test?t=1", "#?t=1", "?t=1#", "#test", "?t=1", "#", "?", ""}) { + test_get_url_query_file_name_one("", suffix, ""); + test_get_url_query_file_name_one("/", suffix, ""); + test_get_url_query_file_name_one("/a/adasd/", suffix, ""); + test_get_url_query_file_name_one("/a/lklrjetn/", suffix, "adasd.asdas"); + test_get_url_query_file_name_one("/", suffix, "a123asadas"); + test_get_url_query_file_name_one("/", suffix, "\\a\\1\\2\\3\\a\\s\\a\\das"); + } +} diff --git a/libs/tdlib/td/tdutils/test/pq.cpp b/libs/tdlib/td/tdutils/test/pq.cpp new file mode 100644 index 0000000000..5210cc2638 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/pq.cpp @@ -0,0 +1,118 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/tests.h" + +#include "td/utils/BigNum.h" +#include "td/utils/common.h" +#include "td/utils/crypto.h" +#include "td/utils/format.h" +#include "td/utils/logging.h" +#include "td/utils/misc.h" + +#include <algorithm> +#include <limits> +#include <utility> + +REGISTER_TESTS(pq) + +using namespace td; + +#if TD_HAVE_OPENSSL +static bool is_prime(uint64 x) { + for (uint64 d = 2; d < x && d * d <= x; d++) { + if (x % d == 0) { + return false; + } + } + return true; +} + +static std::vector<uint64> gen_primes(uint64 L, uint64 R, int limit = 0) { + std::vector<uint64> res; + for (auto x = L; x <= R && (limit <= 0 || res.size() < static_cast<std::size_t>(limit)); x++) { + if (is_prime(x)) { + res.push_back(x); + } + } + return res; +} + +static std::vector<uint64> gen_primes() { + std::vector<uint64> result; + append(result, gen_primes(1, 100)); + append(result, gen_primes((1ull << 31) - 500000, std::numeric_limits<uint64>::max(), 5)); + append(result, gen_primes((1ull << 32) - 500000, std::numeric_limits<uint64>::max(), 5)); + append(result, gen_primes((1ull << 39) - 500000, std::numeric_limits<uint64>::max(), 1)); + return result; +} + +using PqQuery = std::pair<uint64, uint64>; +static bool cmp(const PqQuery &a, const PqQuery &b) { + return a.first * a.second < b.first * b.second; +} +static std::vector<PqQuery> gen_pq_queries() { + std::vector<PqQuery> res; + auto primes = gen_primes(); + for (auto q : primes) { + for (auto p : primes) { + if (p > q) { + break; + } + res.emplace_back(p, q); + } + } + std::sort(res.begin(), res.end(), cmp); + return res; +} + +static void test_pq(uint64 first, uint64 second) { + BigNum p = BigNum::from_decimal(PSLICE() << first); + BigNum q = BigNum::from_decimal(PSLICE() << second); + + BigNum pq; + BigNumContext context; + BigNum::mul(pq, p, q, context); + std::string pq_str = pq.to_binary(); + + std::string p_str, q_str; + int err = td::pq_factorize(pq_str, &p_str, &q_str); + CHECK(err == 0) << first << " * " << second; + + BigNum p_res = BigNum::from_binary(p_str); + BigNum q_res = BigNum::from_binary(q_str); + + CHECK(p_str == p.to_binary()) << td::tag("got", p_res.to_decimal()) << td::tag("expected", first); + CHECK(q_str == q.to_binary()) << td::tag("got", q_res.to_decimal()) << td::tag("expected", second); +} +#endif + +TEST(CryptoPQ, hands) { + ASSERT_EQ(1ull, td::pq_factorize(0)); + ASSERT_EQ(1ull, td::pq_factorize(1)); + ASSERT_EQ(1ull, td::pq_factorize(2)); + ASSERT_EQ(1ull, td::pq_factorize(3)); + ASSERT_EQ(2ull, td::pq_factorize(4)); + ASSERT_EQ(1ull, td::pq_factorize(5)); + ASSERT_EQ(3ull, td::pq_factorize(7 * 3)); + ASSERT_EQ(179424611ull, td::pq_factorize(179424611ull * 179424673ull)); + +#if TD_HAVE_OPENSSL + test_pq(4294467311, 4294467449); +#endif +} + +#if TD_HAVE_OPENSSL +TEST(CryptoPQ, generated_slow) { + for (int i = 0; i < 100000; i++) { + test_pq(2, 2); + } + auto queries = gen_pq_queries(); + for (auto query : queries) { + test_pq(query.first, query.second); + } +} +#endif
\ No newline at end of file diff --git a/libs/tdlib/td/tdutils/test/variant.cpp b/libs/tdlib/td/tdutils/test/variant.cpp new file mode 100644 index 0000000000..5c5e18d1d8 --- /dev/null +++ b/libs/tdlib/td/tdutils/test/variant.cpp @@ -0,0 +1,75 @@ +// +// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 +// +// Distributed under the Boost Software License, Version 1.0. (See accompanying +// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) +// +#include "td/utils/Slice.h" +#include "td/utils/StringBuilder.h" +#include "td/utils/tests.h" +#include "td/utils/Variant.h" + +REGISTER_TESTS(variant); + +using namespace td; + +static const size_t BUF_SIZE = 1024 * 1024; +static char buf[BUF_SIZE], buf2[BUF_SIZE]; +static StringBuilder sb(MutableSlice(buf, BUF_SIZE - 1)); +static StringBuilder sb2(MutableSlice(buf2, BUF_SIZE - 1)); + +static std::string move_sb() { + auto res = sb.as_cslice().str(); + sb.clear(); + return res; +} + +static std::string name(int id) { + if (id == 1) { + return "A"; + } + if (id == 2) { + return "B"; + } + if (id == 3) { + return "C"; + } + return ""; +} + +template <int id> +class Class { + public: + Class() { + sb << "+" << name(id); + } + Class(const Class &) = delete; + Class &operator=(const Class &) = delete; + Class(Class &&) = delete; + Class &operator=(Class &&) = delete; + ~Class() { + sb << "-" << name(id); + } +}; + +using A = Class<1>; +using B = Class<2>; +using C = Class<3>; + +TEST(Variant, simple) { + { + Variant<std::unique_ptr<A>, std::unique_ptr<B>, std::unique_ptr<C>> abc; + ASSERT_STREQ("", sb.as_cslice()); + abc = std::make_unique<A>(); + ASSERT_STREQ("+A", sb.as_cslice()); + sb.clear(); + abc = std::make_unique<B>(); + ASSERT_STREQ("+B-A", sb.as_cslice()); + sb.clear(); + abc = std::make_unique<C>(); + ASSERT_STREQ("+C-B", sb.as_cslice()); + sb.clear(); + } + ASSERT_STREQ("-C", move_sb()); + sb.clear(); +}; |