diff options
Diffstat (limited to 'libs/tdlib/td/test')
-rw-r--r-- | libs/tdlib/td/test/CMakeLists.txt | 50 | ||||
-rw-r--r-- | libs/tdlib/td/test/TestsRunner.cpp | 63 | ||||
-rw-r--r-- | libs/tdlib/td/test/TestsRunner.h | 19 | ||||
-rw-r--r-- | libs/tdlib/td/test/data.cpp | 69 | ||||
-rw-r--r-- | libs/tdlib/td/test/data.h | 15 | ||||
-rw-r--r-- | libs/tdlib/td/test/db.cpp | 575 | ||||
-rw-r--r-- | libs/tdlib/td/test/fuzz_url.cpp | 33 | ||||
-rw-r--r-- | libs/tdlib/td/test/http.cpp | 373 | ||||
-rw-r--r-- | libs/tdlib/td/test/main.cpp | 40 | ||||
-rw-r--r-- | libs/tdlib/td/test/message_entities.cpp | 529 | ||||
-rw-r--r-- | libs/tdlib/td/test/mtproto.cpp | 347 | ||||
-rw-r--r-- | libs/tdlib/td/test/secret.cpp | 1056 | ||||
-rw-r--r-- | libs/tdlib/td/test/string_cleaning.cpp | 107 | ||||
-rw-r--r-- | libs/tdlib/td/test/tdclient.cpp | 837 | ||||
-rw-r--r-- | libs/tdlib/td/test/tests_runner.cpp | 18 | ||||
-rw-r--r-- | libs/tdlib/td/test/tests_runner.h | 18 |
16 files changed, 4149 insertions, 0 deletions
diff --git a/libs/tdlib/td/test/CMakeLists.txt b/libs/tdlib/td/test/CMakeLists.txt new file mode 100644 index 0000000000..d120d8d3fb --- /dev/null +++ b/libs/tdlib/td/test/CMakeLists.txt @@ -0,0 +1,50 @@ +cmake_minimum_required(VERSION 3.0.2 FATAL_ERROR) + +#SOURCE SETS +set(TD_TEST_SOURCE + ${CMAKE_CURRENT_SOURCE_DIR}/db.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/http.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/mtproto.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/message_entities.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/secret.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/string_cleaning.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/TestsRunner.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/tests_runner.cpp + + ${CMAKE_CURRENT_SOURCE_DIR}/TestsRunner.h + ${CMAKE_CURRENT_SOURCE_DIR}/tests_runner.h + + ${CMAKE_CURRENT_SOURCE_DIR}/data.cpp + ${CMAKE_CURRENT_SOURCE_DIR}/data.h + + ${TDUTILS_TEST_SOURCE} + ${TDACTOR_TEST_SOURCE} +) +set(TD_TEST_SOURCE ${TD_TEST_SOURCE} PARENT_SCOPE) + +set(TESTS_MAIN + main.cpp +) + +add_library(all_tests STATIC ${TD_TEST_SOURCE}) +target_include_directories(all_tests PUBLIC $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}>) +target_link_libraries(all_tests PRIVATE tdactor tddb tdcore tdnet tdutils) + +if (NOT CMAKE_CROSSCOMPILING OR EMSCRIPTEN) + #Tests + add_executable(run_all_tests ${TESTS_MAIN} ${TD_TEST_SOURCE}) + if (CLANG AND NOT CYGWIN AND NOT EMSCRIPTEN) + target_compile_options(run_all_tests PUBLIC -fsanitize=undefined -fno-sanitize=vptr) + set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fsanitize=undefined -fno-sanitize=vptr") + endif() + target_include_directories(run_all_tests PUBLIC $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}>) + target_link_libraries(run_all_tests PRIVATE tdactor tddb tdcore tdnet tdutils) + + if (CLANG) +# add_executable(fuzz_url fuzz_url.cpp) +# target_link_libraries(fuzz_url PRIVATE tdclient) +# target_compile_options(fuzz_url PRIVATE "-fsanitize-coverage=trace-pc-guard") + endif() + + add_test(run_all_tests run_all_tests) +endif() diff --git a/libs/tdlib/td/test/TestsRunner.cpp b/libs/tdlib/td/test/TestsRunner.cpp new file mode 100644 index 0000000000..fbe155738e --- /dev/null +++ b/libs/tdlib/td/test/TestsRunner.cpp @@ -0,0 +1,63 @@ +// +// 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 "test/TestsRunner.h" + +#include "td/utils/common.h" +#include "td/utils/FileLog.h" +#include "td/utils/format.h" +#include "td/utils/logging.h" +#include "td/utils/port/path.h" +#include "td/utils/tests.h" + +#include <limits> + +DESC_TESTS(string_cleaning); +DESC_TESTS(message_entities); +DESC_TESTS(variant); +DESC_TESTS(secret); +DESC_TESTS(actors_main); +DESC_TESTS(actors_simple); +DESC_TESTS(actors_workers); +DESC_TESTS(db); +DESC_TESTS(json); +DESC_TESTS(http); +DESC_TESTS(heap); +DESC_TESTS(pq); +DESC_TESTS(mtproto); + +namespace td { + +void TestsRunner::run_all_tests() { + LOAD_TESTS(string_cleaning); + LOAD_TESTS(message_entities); + LOAD_TESTS(variant); + LOAD_TESTS(secret); + LOAD_TESTS(actors_main); + LOAD_TESTS(actors_simple); + LOAD_TESTS(actors_workers); + LOAD_TESTS(db); + LOAD_TESTS(json); + LOAD_TESTS(http); + LOAD_TESTS(heap); + LOAD_TESTS(pq); + LOAD_TESTS(mtproto); + Test::run_all(); +} + +static FileLog file_log; +static TsLog ts_log(&file_log); + +void TestsRunner::init(string dir) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(WARNING)); + chdir(dir).ensure(); + LOG(WARNING) << "Redirect log into " << tag("file", dir + TD_DIR_SLASH + "log.txt"); + if (file_log.init("log.txt", std::numeric_limits<int64>::max())) { + log_interface = &ts_log; + } +} + +} // namespace td diff --git a/libs/tdlib/td/test/TestsRunner.h b/libs/tdlib/td/test/TestsRunner.h new file mode 100644 index 0000000000..a5bc66d855 --- /dev/null +++ b/libs/tdlib/td/test/TestsRunner.h @@ -0,0 +1,19 @@ +// +// 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) +// +#pragma once + +#include "td/utils/common.h" + +namespace td { + +class TestsRunner { + public: + static void init(string dir); + static void run_all_tests(); +}; + +} // namespace td diff --git a/libs/tdlib/td/test/data.cpp b/libs/tdlib/td/test/data.cpp new file mode 100644 index 0000000000..a57a9147c5 --- /dev/null +++ b/libs/tdlib/td/test/data.cpp @@ -0,0 +1,69 @@ +// +// 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 "test/data.h" +namespace td { +static const char thumbnail_arr[] = + "_9j_4AAQSkZJRgABAQEASABIAAD_2wBDAAICAgICAQICAgIDAgIDAwYEAwMDAwcFBQQGCAcJCAgHCAgJCg0LCQoMCggICw8LDA0ODg8OCQsQERAOEQ" + "0ODg7_2wBDAQIDAwMDAwcEBAcOCQgJDg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg4ODg7_wAARCAAyADIDASIA" + "AhEBAxEB_8QAHAAAAQUBAQEAAAAAAAAAAAAAAAUGBwgJAgQD_8QAMRAAAgEDAwMDAgUDBQAAAAAAAQIDBAURAAYSByExCBNBFCIVMlFhcZGhsQkjUm" + "KB_8QAGgEAAgMBAQAAAAAAAAAAAAAABgcAAggEBf_EACgRAAECBgIBBAMBAQAAAAAAAAECEQADBAUhMRJBBhMyUWEUIlJCkf_aAAwDAQACEQMRAD8A" + "381CvUXrbtzYtTLa6ZPx3cKj76SGQLHTn491-_E_9QC38a464dSH2H08gpLZMI9x3dmhom7EwKB98uP2yAP3Ofg6z2M8jVXvM5qqkyFpeTsCCR3Lk9" + "y2eWe_yD3znTG8esEqtR-TVAlJ9qRgqbZJ6T1956ELjyG_T6RRpqQhJHvWrIS-gkf6UfjrGHMTVePUH1HuVYxp7rTWOFj9sVDRocfoOUgYk_00nUXX" + "XqdQTrIdzmsQgHhWUcLowOcdwoPwfB1W3fG8KjaNlpnoqNK-veCWVYXk9qL26eMSOXkPZclUUdxkk-QMaiPZPqFp9wdT7btC92iltlRcZvpqea33Na" + "mNZSuVjcY7BsYDA-SMjByD5c2w01V-DMp5aS4HtffypjnXffULxNPeqmjNwlVU1YZRcFtfCSRjfXRIfUa1bG9SFuuddDbt6UUdkqHIVbhTkmmJPjmp" + "-6P-ckfrjVno5ElgSWJ1kjdQyspyGB8EH51kUWDShpFIbAAkQEjsDktk5wAMAD9hq2Pp16lTrdl2BeJzJSyKzWWWRs8Co5NCD8qR3A-CCv6aE7_45T" + "y5SqijSUKTlSNhvlJ-tkPrI0YMLD5DUmYiTVrE2WssmYMEK_lYxvQLbYHYi4-jRo0p4bUZ6-oC7y3P1IXajJ5QW-lio4jyIKgqsreDg_cwPceQMY86" + "hQAKoUeAMdzk6lzrhb5aT1R7kUqSKpoaiLt-YNEo_wAqw_8ANMuqsVbTWy3x1tEbeXV2E0isWkOcn7QPABAyTnPxp9zfJPH_ABO1US6xYSJwSMZIHE" + "krIyriFYJA2obyIzbMst_8oulZLpwVeiVkA4BPIAIBwORTkAnQOsExZNddoWn1T9Kju6G1yU9bLW01ILraKetiqKgLDJHSH3gVi94hoxIBkOYwCM6r" + "x6yLDYdt9fOi9T03l3DfN2_h8VRPBdYF-uWp-pWSCmZY40CyZYqYuPJOSqQp-0TtuK1dPt0b6oLNvfZg3R9HDM8UU8X3orcQxTDqAGwnEk4J8kAEa8" + "th2NtTZnVjprfZ62fd9vgsMU9jmu1OZJ7YaZ-ApqFsKEVXYq0vEBnZeJIw5UdRfrd5Fd6mptyipCVMFMwUyQXDsWwWcAk6Goc1ts9dZLNT01enishy" + "PcQ6jgs40csSAHc4MPynrYKyxwXHgaenngWfhMvExBl5cWB8Fc4IPgg6WbJdqi07ntd5pJ-Ro6mGpgfOSntsGwD-hGRj9zpSHTjfm-t6XWt2zXWqhg" + "J-rqqGpo-VNLKMMY4nVwW5sCzKue5YjAPHTbobXf6eWjs9-t0duv1RHFIaSGpWYAT_AJMEAEEkkFWAYEeMYJd9vvtHeJyKWanPByT_AEBkD6ZzCUuX" + "j1fYKeZVypmCviAM_ofaTjB5MPrbuRGucUizU0c0bZR1DL2-CMjRriliFNbKanzn2olTP8DGjWczvGo0onQfcV666bXhFwsXUGKnhq6m0LItRSTSGN" + "KpUR5olZgDxAdCCcHCuT8Y1nd0O3dW9X-gta1nv0fTvct_u9xZZ4LuK-sjZgrSSokhb2hyBVEAwBGDjuQdft02GPcuxLjZ3dYnniIhlePmI5B-Vivy" + "AfIyMjI-dUZpukO99qbiEtv6YxVFQjk-9QVkP08mD5BGJMHzhhkds_roG8lNVUKlOhSwkJCSHPEJUVMwBcEqJDgx7NqlU9P6hQySslSgeyQA-W6SIb" + "O6tqQbUk2jcEu0k93WrRPrGjVZPd4_mRhjv7gRyvEqCMnHjTT6f1VmrfUttCyb2RL7FT0sFos9NV0iJTQUsCPUksVHeV5QWLcTyZBnHwp9RoKy47tp" + "pr5a7haK6mOGiniaBBjHZc_GRnlnP9iI1u1huF9r7dV2GrnpNx0NSs9BUUqM5Dg5AKr3Iz_Tv8Egnvg_h12Rbq2pXI9JC-BlBRZSikK5nJ_UEEAOzk" + "dCF_5N5hbJVzpKZM4rKeYmcQ6U8m4aDqKSC_F2B7OIsX1grfTIfVHsSh6gRXPZW5LeJLlYLxaqeWgt0pMyoyTVKf7QkZ4lAVuLEHAbDYMc7F3jsHdX" + "-tXPtXb9FX38PRrW1N5p0MlKwp6f3qcyOo4BFaQxA5yWVVxjSjvDp31r6rdNVgvHS2WgqaWZaumd6yn9mr44WSLizLKnuAI6lk7PChPbOrhem3phH0" + "k9Ie1NpyWZbHdVgM9ypRULMY5nYkoZFGG4jio8-PJ8nsE2bSLJQeK8j_oIPz0SNwRLpqKsokKUrk5H6l2YEKSrQ7AwexE76NGjXmx0QaMDRo1IkfKS" + "KKZCk0ayp_xdQR_fXEVLS0yn6emip8-fbjC_40aNWc66ijDbZj0YGjRo1WLwaNGjUiR__9k"; + +const char *thumbnail = thumbnail_arr; +const size_t thumbnail_size = sizeof(thumbnail_arr) - 1; + +static const char gzip_bomb_arr[] = + "eJzt3VlQU2cYxvGTSFSkqHWJuIHTUqotAetIBmWzVUeUqlRrFcIqZlxCUKRshZBULVS0oqLiwqJWUVwAQQNBgSgt1EHFlS1AQCsZ1qAsASKTRonnnJ" + "ledHrViz5cfvMOOd_v_d-eJCK5vf6MaUmDLK7Jk_tZ9sSDEyYmdo7fOMeqdKSs-jXRxA_m8_fZPGTMW-Rhlf-AP7XxrGTW-qCOyWPtB90qdo-" + "NHtXfEi3Wlul0bWrxYNa1QaVY12ure9OdRgz_SVzyX-" + "nS59eYP1uYV68zHBLi7xcf2qyd7ddSbhX2geFsfPs3aeO04xhKLWffzBGGQyOHIeckkUmXZ1qZoH2a4XBWddSKRKGJ-" + "9UQxe8HGIbDXaZLU4U8SfDf_-UhiXroGG0wJSatvIgVcbXihV2YKTnpdP6ulHWxUtaxsez9U96O_ahCypHn9_AnOow0nI3uu1gRN-" + "Qq18SU0wZXFVfwlcxblapX025Sk6mPNymY37bVhV0LeK_hp7-3gu3nNZizl7pj1FNVSYxlY1QxnzZ4QanK2c7waeuO-" + "MWbmly9NUfIuBQadLP6C8PZouwzf-QKTebaiGp-o-7o1Nrt7zKhq1CZQxt8og6pySOyeiNvlWaSk4UX5tdICf12vNvNSF_" + "3P2ukLLOkokDaHc17B7ItjPsbtnvQBls0snaepGrQ0SegipxsWHsykCfZlqKwoTYROPt1oIK5382XS7tjwmB0dYJRtDrPmrayHq1XH0cuEJlnXROQk" + "-o5K6WcRTceCKdQm-BOipQKGaXnbU_R7rhM5Ny-" + "jFmk4bGpwbBBkY2W3XjDOaGq2o6c1ByxUrBPe6qktNqmxzsqpMSLx0kraXf81dmi72NkOTyILJElskSWBLJElgSyRJbDg8gSWSJLZEkgS2RJIEtkOT" + "yILJElskSWBLJElgSyRJbDi0CWyBJZIktkiSyH74gskSWyRJYEskSWBLJElu_uiCyRJbJElgSyRJbDvsgSWSJLZIkskeXwJLJElsgSWSJLZPluEFn-" + "11nq_NUHhOP3MGfHl7pe3eBeYz_S1Jhbye4pZRxnTCxxbZeP9Jw3gnXUOLb4jm1MyAOjcEmSo-ESnr5HI-2Nj9_kupErWfL5sYyx-" + "phmrlrZSlZXqylU6gsrH5TVryVZHqWWLdY_hU9x2cDJhe8f7efK9XzzVczmzPtrqMF4_qUS_" + "ebj1iYWkHN9InW4gt2YExOkeUru6brq1UH951Spg4vSyY2e7V13Ii5RIhPk0gY3nmjNZplVNT09R-7zrEiprZey8q-kKZxCSdSOSMdN-s_" + "pjUrOyiD5t4jWNTdZ-nXWWlODVzekLwxklIba22WQ-FuUmpg8IcOmTRW2mhKKMh-" + "nbyRiyFwlaCV7ACUoQQlKUIISlKAEJShBCUpQghKUoAQlKEEJSlCCEpSgBCUoQQlKUIISlKAEJShBCUpQghKUoAQlKEEJSlCCEpSgBCUoQQlKUIISl" + "KAEJShBCcp_ouzPDS_" + "jjv6KMWnp4tNs6yuBM3bHLpkuGBP1nLgnMfa3DN8xwjrl9p14o26PpinFCzbvcpC7meuhEyLZUuJCaJGU9nmanev3cOQBLQeSaHfIyLjcYdno0LOUW" + "khtk2bB9revxtcp15Cvxj9qCEj9iSPfmpIlpE0W1k32N-naUPDiE-oOwm2t2xyM-8eKDlELCesXPVC8_XoFTd5T8usVrmuOb_" + "Y32eGpEpyiTdqGL61mOh7rjFhO5XDqZEHtKmbRYt9z1EJmxqR1S99-F8IQJ5T8LoQOp2aPamZy5UAtbXX15d4fSomE8w4-" + "tBzS0zOliRL1wyk0oDil6pa-" + "pbYQsZs31SwoQQlKUIISlKAEJShBCUpQghKUoAQlKEEJSlCCEpSgBCUoQQlKUIISlKAEJShBCUpQghKUoAQlKEEJSlCCEpSgBCUoQQlKUIISlKAEJS" + "hBCUpQ_jtKiTgqPPDO3MOL_Pv3TxL68UbFJ5tlxpXs_XLFDAd3Gy-JIPvL_" + "KOXG414uxZmy1zt7B9axXrM53w9d0a0L0vMGGduuE7ODydO5bOO39kQKiPf5d-Udqb6Ck8SEOGbKSMXsSmvXuqVR1T4ireEkA9010f1xDnVKNpCMy_" + "8PeXBKz7NVkGM9LnfeXeS7_LXbJ7NbWM3mjUkCTrJRdQI8zhVHHl3lUVtHbnbnKqB6wmfMpRlhfdISsvLSbK9PMmc_euSyN-sd-2bGn-_" + "gGUfq3l5kvyhddcwhbA81KTrTW8cl3qg9lDnjhfL5ZpgW9qTt_KDnrP9xgQ8e0m-y39YFLsxN4hxoEkkayYXcbg-" + "TLGznukoHjpzn3qgqJTV2ogJXaK6ctqTgxKUoPyfUeqmx4szeRJuj3josnbUu0O57kcXXX-bi25IozxSONDzF5jalI4"; +const char *gzip_bomb = gzip_bomb_arr; +const size_t gzip_bomb_size = sizeof(gzip_bomb_arr) - 1; +} // namespace td diff --git a/libs/tdlib/td/test/data.h b/libs/tdlib/td/test/data.h new file mode 100644 index 0000000000..c447d5cba6 --- /dev/null +++ b/libs/tdlib/td/test/data.h @@ -0,0 +1,15 @@ +// +// 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) +// +#pragma once +#include "td/utils/common.h" +namespace td { +extern const char *thumbnail; +extern const size_t thumbnail_size; + +extern const char *gzip_bomb; +extern const size_t gzip_bomb_size; +} // namespace td diff --git a/libs/tdlib/td/test/db.cpp b/libs/tdlib/td/test/db.cpp new file mode 100644 index 0000000000..8917dd65b8 --- /dev/null +++ b/libs/tdlib/td/test/db.cpp @@ -0,0 +1,575 @@ +// +// 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/db/binlog/BinlogHelper.h" +#include "td/db/BinlogKeyValue.h" +#include "td/db/SeqKeyValue.h" +#include "td/db/SqliteKeyValue.h" +#include "td/db/SqliteKeyValueSafe.h" +#include "td/db/TsSeqKeyValue.h" + +#include "td/utils/common.h" +#include "td/utils/logging.h" +#include "td/utils/port/FileFd.h" +#include "td/utils/port/thread.h" +#include "td/utils/Random.h" +#include "td/utils/Slice.h" +#include "td/utils/Status.h" +#include "td/utils/tests.h" + +#include <limits> +#include <map> +#include <memory> + +REGISTER_TESTS(db); + +using namespace td; + +template <class ContainerT> +static typename ContainerT::value_type &rand_elem(ContainerT &cont) { + CHECK(0 < cont.size() && cont.size() <= static_cast<size_t>(std::numeric_limits<int>::max())); + return cont[Random::fast(0, static_cast<int>(cont.size()) - 1)]; +} + +TEST(DB, binlog_encryption) { + CSlice binlog_name = "test_binlog"; + Binlog::destroy(binlog_name).ignore(); + + auto hello = DbKey::raw_key(std::string(32, 'A')); + auto cucumber = DbKey::password("cucumber"); + auto empty = DbKey::empty(); + auto long_data = string(10000, 'Z'); + { + Binlog binlog; + binlog.init(binlog_name.str(), [](const BinlogEvent &x) {}).ensure(); + binlog.add_raw_event(BinlogEvent::create_raw(binlog.next_id(), 1, 0, create_storer("AAAA"))); + binlog.add_raw_event(BinlogEvent::create_raw(binlog.next_id(), 1, 0, create_storer("BBBB"))); + binlog.add_raw_event(BinlogEvent::create_raw(binlog.next_id(), 1, 0, create_storer(long_data))); + LOG(INFO) << "SET PASSWORD"; + binlog.change_key(cucumber); + binlog.change_key(hello); + LOG(INFO) << "OK"; + binlog.add_raw_event(BinlogEvent::create_raw(binlog.next_id(), 1, 0, create_storer("CCCC"))); + binlog.close().ensure(); + } + + auto add_suffix = [&] { + auto fd = FileFd::open(binlog_name, FileFd::Flags::Write | FileFd::Flags::Append).move_as_ok(); + fd.write("abacabadaba").ensure(); + }; + + add_suffix(); + + { + std::vector<string> v; + LOG(INFO) << "RESTART"; + Binlog binlog; + binlog.init(binlog_name.str(), [&](const BinlogEvent &x) { v.push_back(x.data_.str()); }, hello).ensure(); + CHECK(v == std::vector<string>({"AAAA", "BBBB", long_data, "CCCC"})); + } + + add_suffix(); + + { + std::vector<string> v; + LOG(INFO) << "RESTART"; + Binlog binlog; + auto status = binlog.init(binlog_name.str(), [&](const BinlogEvent &x) { v.push_back(x.data_.str()); }, cucumber); + CHECK(status.is_error()); + } + + add_suffix(); + + { + std::vector<string> v; + LOG(INFO) << "RESTART"; + Binlog binlog; + auto status = + binlog.init(binlog_name.str(), [&](const BinlogEvent &x) { v.push_back(x.data_.str()); }, cucumber, hello); + CHECK(v == std::vector<string>({"AAAA", "BBBB", long_data, "CCCC"})); + } +}; + +TEST(DB, sqlite_lfs) { + string path = "test_sqlite_db"; + SqliteDb::destroy(path).ignore(); + SqliteDb db; + db.init(path).ensure(); + db.exec("PRAGMA journal_mode=WAL").ensure(); + db.exec("PRAGMA user_version").ensure(); +} + +TEST(DB, sqlite_encryption) { + string path = "test_sqlite_db"; + SqliteDb::destroy(path).ignore(); + + auto empty = DbKey::empty(); + auto cucumber = DbKey::password("cucumber"); + auto tomato = DbKey::raw_key(string(32, 'a')); + + { + auto db = SqliteDb::open_with_key(path, empty).move_as_ok(); + db.set_user_version(123); + auto kv = SqliteKeyValue(); + kv.init_with_connection(db.clone(), "kv"); + kv.set("a", "b"); + } + SqliteDb::open_with_key(path, cucumber).ensure_error(); // key was set... + + SqliteDb::change_key(path, cucumber, empty).ensure(); + + SqliteDb::open_with_key(path, tomato).ensure_error(); + { + auto db = SqliteDb::open_with_key(path, cucumber).move_as_ok(); + auto kv = SqliteKeyValue(); + kv.init_with_connection(db.clone(), "kv"); + CHECK(kv.get("a") == "b"); + CHECK(db.user_version().ok() == 123); + } + + SqliteDb::change_key(path, tomato, cucumber).ensure(); + SqliteDb::change_key(path, tomato, cucumber).ensure(); + + SqliteDb::open_with_key(path, cucumber).ensure_error(); + { + auto db = SqliteDb::open_with_key(path, tomato).move_as_ok(); + auto kv = SqliteKeyValue(); + kv.init_with_connection(db.clone(), "kv"); + CHECK(kv.get("a") == "b"); + CHECK(db.user_version().ok() == 123); + } + + SqliteDb::change_key(path, empty, tomato).ensure(); + SqliteDb::change_key(path, empty, tomato).ensure(); + + { + auto db = SqliteDb::open_with_key(path, empty).move_as_ok(); + auto kv = SqliteKeyValue(); + kv.init_with_connection(db.clone(), "kv"); + CHECK(kv.get("a") == "b"); + CHECK(db.user_version().ok() == 123); + } + SqliteDb::open_with_key(path, cucumber).ensure_error(); +} + +using SeqNo = uint64; +struct DbQuery { + enum Type { Get, Set, Erase } type; + SeqNo tid = 0; + int32 id = 0; + string key; + string value; +}; + +template <class ImplT> +class QueryHandler { + public: + ImplT &impl() { + return impl_; + } + void do_query(DbQuery &query) { + switch (query.type) { + case DbQuery::Get: + query.value = impl_.get(query.key); + return; + case DbQuery::Set: + query.tid = impl_.set(query.key, query.value); + return; + case DbQuery::Erase: + query.tid = impl_.erase(query.key); + return; + } + } + + private: + ImplT impl_; +}; + +class SqliteKV { + public: + string get(string key) { + return kv_->get().get(key); + } + SeqNo set(string key, string value) { + kv_->get().set(key, value); + return 0; + } + SeqNo erase(string key) { + kv_->get().erase(key); + return 0; + } + Status init(string name) { + auto sql_connection = std::make_shared<SqliteConnectionSafe>(name); + kv_ = std::make_shared<SqliteKeyValueSafe>("kv", sql_connection); + return Status::OK(); + } + void close() { + kv_.reset(); + } + + private: + std::shared_ptr<SqliteKeyValueSafe> kv_; +}; + +class BaselineKV { + public: + string get(string key) { + return map_[key]; + } + SeqNo set(string key, string value) { + map_[key] = value; + return ++current_tid_; + } + SeqNo erase(string key) { + map_.erase(key); + return ++current_tid_; + } + + private: + std::map<string, string> map_; + SeqNo current_tid_ = 0; +}; + +TEST(DB, key_value) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(INFO)); + std::vector<std::string> keys; + std::vector<std::string> values; + + for (int i = 0; i < 100; i++) { + keys.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + for (int i = 0; i < 1000; i++) { + values.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + + int queries_n = 300000; + std::vector<DbQuery> queries(queries_n); + for (auto &q : queries) { + int op = Random::fast(0, 2); + const auto &key = rand_elem(keys); + const auto &value = rand_elem(values); + if (op == 0) { + q.type = DbQuery::Get; + q.key = key; + } else if (op == 1) { + q.type = DbQuery::Erase; + q.key = key; + } else if (op == 2) { + q.type = DbQuery::Set; + q.key = key; + q.value = value; + } + } + + QueryHandler<BaselineKV> baseline; + QueryHandler<SeqKeyValue> kv; + QueryHandler<TsSeqKeyValue> ts_kv; + QueryHandler<BinlogKeyValue<Binlog>> new_kv; + + CSlice new_kv_name = "test_new_kv"; + Binlog::destroy(new_kv_name).ignore(); + new_kv.impl().init(new_kv_name.str()).ensure(); + + QueryHandler<SqliteKeyValue> sqlite_kv; + CSlice name = "test_sqlite_kv"; + SqliteDb::destroy(name).ignore(); + sqlite_kv.impl().init(name.str()).ensure(); + + int cnt = 0; + for (auto &q : queries) { + DbQuery a = q; + DbQuery b = q; + DbQuery c = q; + DbQuery d = q; + DbQuery e = q; + baseline.do_query(a); + kv.do_query(b); + ts_kv.do_query(c); + sqlite_kv.do_query(d); + new_kv.do_query(e); + ASSERT_EQ(a.value, b.value); + ASSERT_EQ(a.value, c.value); + ASSERT_EQ(a.value, d.value); + ASSERT_EQ(a.value, e.value); + if (cnt++ % 10000 == 0) { + new_kv.impl().init(new_kv_name.str()).ensure(); + } + } +} + +TEST(DB, thread_key_value) { +#if !TD_THREAD_UNSUPPORTED + std::vector<std::string> keys; + std::vector<std::string> values; + + for (int i = 0; i < 100; i++) { + keys.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + for (int i = 0; i < 1000; i++) { + values.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + + int threads_n = 4; + int queries_n = 100000; + + std::vector<std::vector<DbQuery>> queries(threads_n, std::vector<DbQuery>(queries_n)); + for (auto &qs : queries) { + for (auto &q : qs) { + int op = Random::fast(0, 10); + const auto &key = rand_elem(keys); + const auto &value = rand_elem(values); + if (op > 1) { + q.type = DbQuery::Get; + q.key = key; + } else if (op == 0) { + q.type = DbQuery::Erase; + q.key = key; + } else if (op == 1) { + q.type = DbQuery::Set; + q.key = key; + q.value = value; + } + } + } + + QueryHandler<BaselineKV> baseline; + QueryHandler<TsSeqKeyValue> ts_kv; + + std::vector<thread> threads(threads_n); + std::vector<std::vector<DbQuery>> res(threads_n); + for (int i = 0; i < threads_n; i++) { + threads[i] = thread([&ts_kv, &queries, &res, i]() { + for (auto q : queries[i]) { + ts_kv.do_query(q); + res[i].push_back(q); + } + }); + } + for (auto &thread : threads) { + thread.join(); + } + + std::vector<std::size_t> pos(threads_n); + while (true) { + bool was = false; + for (int i = 0; i < threads_n; i++) { + auto p = pos[i]; + if (p == res[i].size()) { + continue; + } + auto &q = res[i][p]; + if (q.tid == 0) { + if (q.type == DbQuery::Get) { + auto nq = q; + baseline.do_query(nq); + if (nq.value == q.value) { + was = true; + pos[i]++; + } + } else { + was = true; + pos[i]++; + } + } + } + if (was) { + continue; + } + + int best = -1; + SeqNo best_tid = 0; + for (int i = 0; i < threads_n; i++) { + auto p = pos[i]; + if (p == res[i].size()) { + continue; + } + was = true; + auto &q = res[i][p]; + if (q.tid != 0) { + if (best == -1 || q.tid < best_tid) { + best = i; + best_tid = q.tid; + } + } + } + if (!was) { + break; + } + ASSERT_TRUE(best != -1); + baseline.do_query(res[best][pos[best]]); + pos[best]++; + } +#endif +} + +TEST(DB, persistent_key_value) { + using KeyValue = BinlogKeyValue<ConcurrentBinlog>; + // using KeyValue = PersistentKeyValue; + // using KeyValue = SqliteKV; + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(WARNING)); + std::vector<std::string> keys; + std::vector<std::string> values; + CSlice name = "test_pmc"; + Binlog::destroy(name).ignore(); + SqliteDb::destroy(name).ignore(); + + for (int i = 0; i < 100; i++) { + keys.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + for (int i = 0; i < 1000; i++) { + values.push_back(rand_string('a', 'b', Random::fast(1, 10))); + } + + QueryHandler<BaselineKV> baseline; + + for (int iter = 0; iter < 25; iter++) { + int threads_n = 4; + int queries_n = 3000 / threads_n; + + std::vector<std::vector<DbQuery>> queries(threads_n, std::vector<DbQuery>(queries_n)); + for (auto &qs : queries) { + for (auto &q : qs) { + int op = Random::fast(0, 10); + const auto &key = rand_elem(keys); + const auto &value = rand_elem(values); + if (op > 1) { + q.type = DbQuery::Get; + q.key = key; + } else if (op == 0) { + q.type = DbQuery::Erase; + q.key = key; + } else if (op == 1) { + q.type = DbQuery::Set; + q.key = key; + q.value = value; + } + } + } + + std::vector<std::vector<DbQuery>> res(threads_n); + class Worker : public Actor { + public: + Worker(ActorShared<> parent, std::shared_ptr<QueryHandler<KeyValue>> kv, const std::vector<DbQuery> *queries, + std::vector<DbQuery> *res) + : parent_(std::move(parent)), kv_(std::move(kv)), queries_(queries), res_(res) { + } + void loop() override { + for (auto q : *queries_) { + kv_->do_query(q); + res_->push_back(q); + } + stop(); + } + + private: + ActorShared<> parent_; + std::shared_ptr<QueryHandler<KeyValue>> kv_; + const std::vector<DbQuery> *queries_; + std::vector<DbQuery> *res_; + }; + class Main : public Actor { + public: + Main(int threads_n, const std::vector<std::vector<DbQuery>> *queries, std::vector<std::vector<DbQuery>> *res) + : threads_n_(threads_n), queries_(queries), res_(res) { + } + + void start_up() override { + LOG(INFO) << "start_up"; + kv_->impl().init("test_pmc").ensure(); + ref_cnt_ = threads_n_; + for (int i = 0; i < threads_n_; i++) { + create_actor_on_scheduler<Worker>("Worker", i + 1, actor_shared(this, 2), kv_, &queries_->at(i), &res_->at(i)) + .release(); + } + } + + void tear_down() override { + LOG(INFO) << "tear_down"; + // kv_->impl().close(); + } + void hangup_shared() override { + LOG(INFO) << "hangup"; + ref_cnt_--; + if (ref_cnt_ == 0) { + kv_->impl().close(); + Scheduler::instance()->finish(); + stop(); + } + } + void hangup() override { + LOG(ERROR) << "BAD HANGUP"; + } + + private: + int threads_n_; + const std::vector<std::vector<DbQuery>> *queries_; + std::vector<std::vector<DbQuery>> *res_; + + std::shared_ptr<QueryHandler<KeyValue>> kv_{new QueryHandler<KeyValue>()}; + int ref_cnt_; + }; + + ConcurrentScheduler sched; + sched.init(threads_n); + sched.create_actor_unsafe<Main>(0, "Main", threads_n, &queries, &res).release(); + sched.start(); + while (sched.run_main(10)) { + // empty + } + sched.finish(); + + std::vector<std::size_t> pos(threads_n); + while (true) { + bool was = false; + for (int i = 0; i < threads_n; i++) { + auto p = pos[i]; + if (p == res[i].size()) { + continue; + } + auto &q = res[i][p]; + if (q.tid == 0) { + if (q.type == DbQuery::Get) { + auto nq = q; + baseline.do_query(nq); + if (nq.value == q.value) { + was = true; + pos[i]++; + } + } else { + was = true; + pos[i]++; + } + } + } + if (was) { + continue; + } + + int best = -1; + SeqNo best_tid = 0; + for (int i = 0; i < threads_n; i++) { + auto p = pos[i]; + if (p == res[i].size()) { + continue; + } + was = true; + auto &q = res[i][p]; + if (q.tid != 0) { + if (best == -1 || q.tid < best_tid) { + best = i; + best_tid = q.tid; + } + } + } + if (!was) { + break; + } + ASSERT_TRUE(best != -1); + baseline.do_query(res[best][pos[best]]); + pos[best]++; + } + } +} diff --git a/libs/tdlib/td/test/fuzz_url.cpp b/libs/tdlib/td/test/fuzz_url.cpp new file mode 100644 index 0000000000..74047135c0 --- /dev/null +++ b/libs/tdlib/td/test/fuzz_url.cpp @@ -0,0 +1,33 @@ +// +// 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/telegram/MessageEntity.h" + +#include "td/utils/common.h" +#include "td/utils/logging.h" +#include "td/utils/Slice.h" + +#include <cstddef> +#include <cstdint> + +static td::string get_utf_string(td::Slice from) { + td::string res; + td::string alph = " ab@./01#"; + for (auto c : from) { + res += alph[td::uint8(c) % alph.size()]; + } + LOG(ERROR) << res; + return res; +} + +extern "C" int LLVMFuzzerTestOneInput(std::uint8_t *data, std::size_t data_size) { + td::find_urls(get_utf_string(td::Slice(data, data_size))); + //td::find_hashtags(get_utf_string(td::Slice(data, data_size))); + //td::find_bot_commands(get_utf_string(td::Slice(data, data_size))); + //td::is_email_address(get_utf_string(td::Slice(data, data_size))); + //td::find_mentions(get_utf_string(td::Slice(data, data_size))); + return 0; +} diff --git a/libs/tdlib/td/test/http.cpp b/libs/tdlib/td/test/http.cpp new file mode 100644 index 0000000000..98c94b2e8a --- /dev/null +++ b/libs/tdlib/td/test/http.cpp @@ -0,0 +1,373 @@ +// +// 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/net/HttpChunkedByteFlow.h" +#include "td/net/HttpHeaderCreator.h" +#include "td/net/HttpQuery.h" +#include "td/net/HttpReader.h" + +#include "td/utils/AesCtrByteFlow.h" +#include "td/utils/base64.h" +#include "td/utils/buffer.h" +#include "td/utils/BufferedFd.h" +#include "td/utils/ByteFlow.h" +#include "td/utils/crypto.h" +#include "td/utils/format.h" +#include "td/utils/Gzip.h" +#include "td/utils/GzipByteFlow.h" +#include "td/utils/logging.h" +#include "td/utils/misc.h" +#include "td/utils/port/Fd.h" +#include "td/utils/port/FileFd.h" +#include "td/utils/port/path.h" +#include "td/utils/port/thread_local.h" +#include "td/utils/Random.h" +#include "td/utils/Slice.h" +#include "td/utils/Status.h" + +#include "test/data.h" + +#include <algorithm> +#include <cstdlib> +#include <limits> + +REGISTER_TESTS(http) + +using namespace td; + +static string make_chunked(string str) { + auto v = rand_split(str); + string res; + for (auto &s : v) { + res += PSTRING() << format::as_hex_dump(int(s.size())); + res += "\r\n"; + res += s; + res += "\r\n"; + } + res += "0\r\n\r\n"; + return res; +} + +static string gen_http_content() { + int t = Random::fast(0, 2); + int len; + if (t == 0) { + len = Random::fast(1, 10); + } else if (t == 1) { + len = Random::fast(100, 200); + } else { + len = Random::fast(1000, 20000); + } + return rand_string(std::numeric_limits<char>::min(), std::numeric_limits<char>::max(), len); +} + +static string make_http_query(string content, bool is_chunked, bool is_gzip, double gzip_k = 5, + string zip_override = "") { + HttpHeaderCreator hc; + hc.init_post("/"); + hc.add_header("jfkdlsahhjk", rand_string('a', 'z', Random::fast(1, 2000))); + if (is_gzip) { + BufferSlice zip; + if (zip_override.empty()) { + zip = gzencode(content, gzip_k); + } else { + zip = BufferSlice(zip_override); + } + if (!zip.empty()) { + hc.add_header("content-encoding", "gzip"); + content = zip.as_slice().str(); + } + } + if (is_chunked) { + hc.add_header("transfer-encoding", "chunked"); + content = make_chunked(content); + } else { + hc.set_content_size(content.size()); + } + string res; + auto r_header = hc.finish(); + CHECK(r_header.is_ok()); + res += r_header.ok().str(); + res += content; + return res; +} + +static string rand_http_query(string content) { + bool is_chunked = Random::fast(0, 1) == 0; + bool is_gzip = Random::fast(0, 1) == 0; + return make_http_query(std::move(content), is_chunked, is_gzip); +} + +static string join(const std::vector<string> &v) { + string res; + for (auto &s : v) { + res += s; + } + return res; +} + +TEST(Http, stack_overflow) { + ChainBufferWriter writer; + BufferSlice slice(string(256, 'A')); + for (int i = 0; i < 1000000; i++) { + ChainBufferWriter tmp_writer; + writer.append(slice.clone()); + } + { + auto reader = writer.extract_reader(); + reader.sync_with_writer(); + } +} + +TEST(Http, reader) { +#if TD_ANDROID || TD_TIZEN + return; +#endif + clear_thread_locals(); + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(INFO)); + auto start_mem = BufferAllocator::get_buffer_mem(); + { + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + HttpReader reader; + int max_post_size = 10000; + reader.init(&input, max_post_size, 0); + + std::srand(4); + std::vector<string> contents(1000); + std::generate(contents.begin(), contents.end(), gen_http_content); + auto v = td::transform(contents, rand_http_query); + auto vec_str = rand_split(join(v)); + + HttpQuery q; + std::vector<string> res; + for (auto &str : vec_str) { + input_writer.append(str); + input.sync_with_writer(); + while (true) { + auto r_state = reader.read_next(&q); + LOG_IF(ERROR, r_state.is_error()) << r_state.error() << tag("ok", res.size()); + ASSERT_TRUE(r_state.is_ok()); + auto state = r_state.ok(); + if (state == 0) { + if (q.files_.empty()) { + ASSERT_TRUE(td::narrow_cast<int>(q.content_.size()) <= max_post_size); + auto expected = contents[res.size()]; + ASSERT_EQ(expected, q.content_.str()); + res.push_back(q.content_.str()); + } else { + auto r_fd = FileFd::open(q.files_[0].temp_file_name, FileFd::Read); + ASSERT_TRUE(r_fd.is_ok()); + auto fd = r_fd.move_as_ok(); + string content(td::narrow_cast<size_t>(q.files_[0].size), '\0'); + auto r_size = fd.read(MutableSlice(content)); + ASSERT_TRUE(r_size.is_ok()); + ASSERT_TRUE(r_size.ok() == content.size()); + ASSERT_TRUE(td::narrow_cast<int>(content.size()) > max_post_size); + ASSERT_EQ(contents[res.size()], content); + res.push_back(content); + fd.close(); + } + } else { + break; + } + } + } + ASSERT_EQ(contents.size(), res.size()); + ASSERT_EQ(contents, res); + } + clear_thread_locals(); + ASSERT_EQ(start_mem, BufferAllocator::get_buffer_mem()); +} + +TEST(Http, gzip_bomb) { +#if TD_ANDROID || TD_TIZEN || TD_EMSCRIPTEN // the test should be disabled on low-memory systems + return; +#endif + auto gzip_bomb_str = + gzdecode(gzdecode(base64url_decode(Slice(gzip_bomb, gzip_bomb_size)).ok()).as_slice()).as_slice().str(); + + auto query = make_http_query("", false, true, 0.01, gzip_bomb_str); + auto parts = rand_split(query); + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + HttpReader reader; + HttpQuery q; + reader.init(&input, 100000000, 0); + for (auto &part : parts) { + input_writer.append(part); + input.sync_with_writer(); + auto r_state = reader.read_next(&q); + if (r_state.is_error()) { + LOG(INFO) << r_state.error(); + return; + } + ASSERT_TRUE(r_state.ok() != 0); + } +} + +TEST(Http, aes_ctr_encode_decode_flow) { + auto str = rand_string('a', 'z', 1000000); + auto parts = rand_split(str); + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + ByteFlowSource source(&input); + UInt256 key; + UInt128 iv; + Random::secure_bytes(key.raw, sizeof(key)); + Random::secure_bytes(iv.raw, sizeof(iv)); + AesCtrByteFlow aes_encode; + aes_encode.init(key, iv); + AesCtrByteFlow aes_decode; + aes_decode.init(key, iv); + ByteFlowSink sink; + source >> aes_encode >> aes_decode >> sink; + + ASSERT_TRUE(!sink.is_ready()); + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + ASSERT_TRUE(!sink.is_ready()); + source.close_input(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()); +} + +TEST(Http, aes_file_encryption) { + auto str = rand_string('a', 'z', 1000000); + CSlice name = "test_encryption"; + unlink(name).ignore(); + UInt256 key; + UInt128 iv; + Random::secure_bytes(key.raw, sizeof(key)); + Random::secure_bytes(iv.raw, sizeof(iv)); + + { + BufferedFdBase<FileFd> fd(FileFd::open(name, FileFd::Write | FileFd::Create).move_as_ok()); + + auto parts = rand_split(str); + + ChainBufferWriter output_writer; + auto output_reader = output_writer.extract_reader(); + ByteFlowSource source(&output_reader); + AesCtrByteFlow aes_encode; + aes_encode.init(key, iv); + ByteFlowSink sink; + + source >> aes_encode >> sink; + fd.set_output_reader(sink.get_output()); + + for (auto &part : parts) { + output_writer.append(part); + source.wakeup(); + fd.flush_write().ensure(); + } + fd.close(); + } + + { + BufferedFdBase<FileFd> fd(FileFd::open(name, FileFd::Read).move_as_ok()); + + ChainBufferWriter input_writer; + auto input_reader = input_writer.extract_reader(); + ByteFlowSource source(&input_reader); + AesCtrByteFlow aes_encode; + aes_encode.init(key, iv); + ByteFlowSink sink; + source >> aes_encode >> sink; + fd.set_input_writer(&input_writer); + + fd.update_flags(Fd::Flag::Read); + while (can_read(fd)) { + fd.flush_read(4096).ensure(); + source.wakeup(); + } + + fd.close(); + + source.close_input(Status::OK()); + ASSERT_TRUE(sink.is_ready()); + LOG_IF(ERROR, sink.status().is_error()) << sink.status(); + ASSERT_TRUE(sink.status().is_ok()); + auto result = sink.result()->move_as_buffer_slice().as_slice().str(); + ASSERT_EQ(str, result); + } +} + +TEST(Http, chunked_flow) { + auto str = rand_string('a', 'z', 100); + auto parts = rand_split(make_chunked(str)); + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + ByteFlowSource source(&input); + HttpChunkedByteFlow chunked_flow; + ByteFlowSink sink; + source >> chunked_flow >> sink; + + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + source.close_input(Status::OK()); + ASSERT_TRUE(sink.is_ready()); + LOG_IF(ERROR, sink.status().is_error()) << sink.status(); + ASSERT_TRUE(sink.status().is_ok()); + auto res = sink.result()->move_as_buffer_slice().as_slice().str(); + ASSERT_EQ(str.size(), res.size()); + ASSERT_EQ(str, res); +} + +TEST(Http, chunked_flow_error) { + auto str = rand_string('a', 'z', 100000); + for (int d = 1; d < 100; d += 10) { + auto new_str = make_chunked(str); + new_str.resize(str.size() - d); + auto parts = rand_split(new_str); + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + ByteFlowSource source(&input); + HttpChunkedByteFlow chunked_flow; + ByteFlowSink sink; + source >> chunked_flow >> sink; + + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + ASSERT_TRUE(!sink.is_ready()); + source.close_input(Status::OK()); + ASSERT_TRUE(sink.is_ready()); + ASSERT_TRUE(!sink.status().is_ok()); + } +} + +TEST(Http, gzip_chunked_flow) { + auto str = rand_string('a', 'z', 1000000); + auto parts = rand_split(make_chunked(gzencode(str).as_slice().str())); + + auto input_writer = ChainBufferWriter::create_empty(); + auto input = input_writer.extract_reader(); + ByteFlowSource source(&input); + HttpChunkedByteFlow chunked_flow; + GzipByteFlow gzip_flow(Gzip::Decode); + ByteFlowSink sink; + source >> chunked_flow >> gzip_flow >> sink; + + for (auto &part : parts) { + input_writer.append(part); + source.wakeup(); + } + source.close_input(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/test/main.cpp b/libs/tdlib/td/test/main.cpp new file mode 100644 index 0000000000..0ef46c75b2 --- /dev/null +++ b/libs/tdlib/td/test/main.cpp @@ -0,0 +1,40 @@ +// +// 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/logging.h" + +#include <cstring> + +#if TD_EMSCRIPTEN +#include <emscripten.h> +#endif + +int main(int argc, char **argv) { + // TODO port OptionsParser to Windows + for (int i = 1; i < argc; i++) { + if (!std::strcmp(argv[i], "--filter")) { + CHECK(i + 1 < argc); + td::Test::add_substr_filter(argv[++i]); + } + if (!std::strcmp(argv[i], "--stress")) { + td::Test::set_stress_flag(true); + } + } +#if TD_EMSCRIPTEN + emscripten_set_main_loop( + [] { + if (!td::Test::run_all_step()) { + emscripten_cancel_main_loop(); + } + }, + 10, 0); +#else + td::Test::run_all(); +#endif + return 0; +} diff --git a/libs/tdlib/td/test/message_entities.cpp b/libs/tdlib/td/test/message_entities.cpp new file mode 100644 index 0000000000..473a87140a --- /dev/null +++ b/libs/tdlib/td/test/message_entities.cpp @@ -0,0 +1,529 @@ +// +// 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/telegram/MessageEntity.h" + +#include "td/utils/format.h" +#include "td/utils/logging.h" +#include "td/utils/tests.h" + +REGISTER_TESTS(message_entities); + +using namespace td; + +static void check_mention(string str, std::vector<string> expected) { + auto result_slice = find_mentions(str); + std::vector<string> result; + for (auto &it : result_slice) { + result.push_back(it.str()); + } + if (result != expected) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result)) + << tag("expected", format::as_array(expected)); + } +} + +TEST(MessageEntities, mention) { + check_mention("@mention", {"@mention"}); + check_mention("@mention ", {"@mention"}); + check_mention(" @mention", {"@mention"}); + check_mention(" @mention ", {"@mention"}); + check_mention("@abc @xyz @abc @xyz @xxx@yyy @ttt", {}); + check_mention("@abcde @xyzxy @abcde @xyzxy @xxxxx@yyyyy @ttttt", + {"@abcde", "@xyzxy", "@abcde", "@xyzxy", "@xxxxx", "@ttttt"}); + check_mention("no@mention", {}); + check_mention("@n", {}); + check_mention("@abcdefghijklmnopqrstuvwxyz123456", {"@abcdefghijklmnopqrstuvwxyz123456"}); + check_mention("@abcdefghijklmnopqrstuvwxyz1234567", {}); + check_mention("нет@mention", {}); + check_mention("@ya @gif @wiki @vid @bing @pic @bold @imdb @coub @like @vote @giff @cap ya cap @y @yar @bingg @bin", + {"@ya", "@gif", "@wiki", "@vid", "@bing", "@pic", "@bold", "@imdb", "@coub", "@like", "@vote", "@giff", + "@cap", "@bingg"}); +}; + +static void check_bot_command(string str, std::vector<string> expected) { + auto result_slice = find_bot_commands(str); + std::vector<string> result; + for (auto &it : result_slice) { + result.push_back(it.str()); + } + if (result != expected) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result)) + << tag("expected", format::as_array(expected)); + } +} + +TEST(MessageEntities, bot_command) { + // 1..64@3..32 + check_bot_command("/abc", {"/abc"}); + check_bot_command(" /abc", {"/abc"}); + check_bot_command("/abc ", {"/abc"}); + check_bot_command(" /abc ", {"/abc"}); + check_bot_command("/a@abc", {"/a@abc"}); + check_bot_command("/a@b", {}); + check_bot_command("/@bfdsa", {}); + check_bot_command("/test/", {}); +} + +static void check_hashtag(string str, std::vector<string> expected) { + auto result_slice = find_hashtags(str); + std::vector<string> result; + for (auto &it : result_slice) { + result.push_back(it.str()); + } + if (result != expected) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result)) + << tag("expected", format::as_array(expected)); + } +} + +TEST(MessageEntities, hashtag) { + check_hashtag("", {}); + check_hashtag("#", {}); + check_hashtag("##", {}); + check_hashtag("###", {}); + check_hashtag("#a", {"#a"}); + check_hashtag(" #a", {"#a"}); + check_hashtag("#a ", {"#a"}); + check_hashtag(" #я ", {"#я"}); + check_hashtag(" я#a ", {}); + check_hashtag(" #a# ", {}); + check_hashtag(" #123 ", {}); + check_hashtag(" #123a ", {"#123a"}); + check_hashtag(" #a123 ", {"#a123"}); + check_hashtag(" #123a# ", {}); + check_hashtag(" #" + string(300, '1'), {}); + check_hashtag(" #" + string(256, '1'), {}); + check_hashtag(" #" + string(256, '1') + "a ", {}); + check_hashtag(" #" + string(255, '1') + "a", {"#" + string(255, '1') + "a"}); + check_hashtag(" #" + string(255, '1') + "Я", {"#" + string(255, '1') + "Я"}); + check_hashtag(" #" + string(255, '1') + "a" + string(255, 'b') + "# ", {}); + check_hashtag("#a#b #c #d", {"#c", "#d"}); + check_hashtag("#test", {"#test"}); + check_hashtag(u8"\U0001F604\U0001F604\U0001F604\U0001F604 \U0001F604\U0001F604\U0001F604#" + string(200, '1') + + "ООО" + string(200, '2'), + {"#" + string(200, '1') + "ООО" + string(53, '2')}); + check_hashtag(u8"#a\u2122", {"#a"}); +} + +static void check_cashtag(string str, std::vector<string> expected) { + auto result_slice = find_cashtags(str); + std::vector<string> result; + for (auto &it : result_slice) { + result.push_back(it.str()); + } + if (result != expected) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result)) + << tag("expected", format::as_array(expected)); + } +} + +TEST(MessageEntities, cashtag) { + check_cashtag("", {}); + check_cashtag("$", {}); + check_cashtag("$$", {}); + check_cashtag("$$$", {}); + check_cashtag("$a", {}); + check_cashtag(" $a", {}); + check_cashtag("$a ", {}); + check_cashtag(" $я ", {}); + check_cashtag("$ab", {}); + check_cashtag("$abc", {}); + check_cashtag("$", {}); + check_cashtag("$A", {}); + check_cashtag("$AB", {}); + check_cashtag("$АBC", {}); + check_cashtag("$АВС", {}); + check_cashtag("$ABC", {"$ABC"}); + check_cashtag("$ABCD", {"$ABCD"}); + check_cashtag("$ABCDE", {"$ABCDE"}); + check_cashtag("$ABCDEF", {"$ABCDEF"}); + check_cashtag("$ABCDEFG", {"$ABCDEFG"}); + check_cashtag("$ABCDEFGH", {"$ABCDEFGH"}); + check_cashtag("$ABCDEFGHJ", {}); + check_cashtag("$ABCDEFGH1", {}); + check_cashtag(" $XYZ", {"$XYZ"}); + check_cashtag("$XYZ ", {"$XYZ"}); + check_cashtag(" $XYZ ", {"$XYZ"}); + check_cashtag(" $$XYZ ", {}); + check_cashtag(" $XYZ$ ", {}); + check_cashtag(" $ABC1 ", {}); + check_cashtag(" $1ABC ", {}); + check_cashtag(" 1$ABC ", {}); + check_cashtag(" А$ABC ", {}); + check_cashtag("$ABC$DEF $GHI $KLM", {"$GHI", "$KLM"}); + check_cashtag("$TEST", {"$TEST"}); + check_cashtag(u8"$ABC\u2122", {"$ABC"}); + check_cashtag(u8"\u2122$ABC", {"$ABC"}); + check_cashtag(u8"\u2122$ABC\u2122", {"$ABC"}); +} + +static void check_is_email_address(string str, bool expected) { + bool result = is_email_address(str); + LOG_IF(FATAL, result != expected) << "Expected " << expected << " as result of is_email_address(" << str << ")"; +} + +TEST(MessageEntities, is_email_address) { + check_is_email_address("telegram.org", false); + check_is_email_address("security@telegram.org", true); + check_is_email_address("security.telegram.org", false); + check_is_email_address("", false); + check_is_email_address("@", false); + check_is_email_address("A@a.a.a.ab", true); + check_is_email_address("A@a.ab", true); + check_is_email_address("Test@aa.aa.aa.aa", true); + check_is_email_address("Test@test.abd", true); + check_is_email_address("a@a.a.a.ab", true); + check_is_email_address("test@test.abd", true); + check_is_email_address("test@test.com", true); + check_is_email_address("test.abd", false); + check_is_email_address("a.ab", false); + check_is_email_address("a.bc@d.ef", true); + + vector<string> bad_userdatas = {"", + "a.a.a.a.a.a.a.a.a.a.a.a", + "+.+.+.+.+.+", + "*.a.a", + "a.*.a", + "a.a.*", + "a.a.", + "a.a.abcdefghijklmnopqrstuvwxyz0123456789", + "a.abcdefghijklmnopqrstuvwxyz0.a", + "abcdefghijklmnopqrstuvwxyz0.a.a"}; + vector<string> good_userdatas = {"a.a.a.a.a.a.a.a.a.a.a", + "a+a+a+a+a+a+a+a+a+a+a", + "+.+.+.+.+._", + "aozAQZ0-5-9_+-aozAQZ0-5-9_.aozAQZ0-5-9_.-._.+-", + "a.a.a", + "a.a.abcdefghijklmnopqrstuvwxyz012345678", + "a.abcdefghijklmnopqrstuvwxyz.a", + "a..a", + "abcdefghijklmnopqrstuvwxyz.a.a", + ".a.a"}; + + vector<string> bad_domains = {"", + ".", + "abc", + "localhost", + "a.a.a.a.a.a.a.ab", + ".......", + "a.a.a.a.a.a+ab", + "a+a.a.a.a.a.ab", + "a.a.a.a.a.a.a", + "a.a.a.a.a.a.abcdefg", + "a.a.a.a.a.a.ab0yz", + "a.a.a.a.a.a.ab9yz", + "a.a.a.a.a.a.ab-yz", + "a.a.a.a.a.a.ab_yz", + "a.a.a.a.a.a.ab*yz", + ".ab", + ".a.ab", + "a..ab", + "a.a.a..a.ab", + ".a.a.a.a.ab", + "abcdefghijklmnopqrstuvwxyz01234.ab", + "ab0cd.abd.aA*sd.0.9.0-9.ABOYZ", + "ab*cd.abd.aAasd.0.9.0-9.ABOYZ", + "ab0cd.abd.aAasd.0.9.0*9.ABOYZ", + "*b0cd.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0c*.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.0-*.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.*-9.ABOYZ", + "-b0cd.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0c-.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.-.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.--9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.0--.ABOYZ", + "_b0cd.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0c_.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd._.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9._-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.0-_.ABOYZ", + "-.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0cd.ab_d.-.0.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9.-.ABOYZ", + "_.ab_d.aA-sd.0.9.0-9.ABOYZ", + "ab0cd.ab_d._.0.9.0-9.ABOYZ", + "ab0cd.ab_d.aA-sd.0.9._.ABOYZ"}; + vector<string> good_domains = {"a.a.a.a.a.a.ab", + "a.a.a.a.a.a.abcdef", + "a.a.a.a.a.a.aboyz", + "a.a.a.a.a.a.ABOYZ", + "a.a.a.a.a.a.AbOyZ", + "abcdefghijklmnopqrstuvwxyz0123.ab", + "ab0cd.ab_d.aA-sd.0.9.0-9.ABOYZ", + "A.Z.aA-sd.a.z.0-9.ABOYZ"}; + + for (auto &userdata : bad_userdatas) { + for (auto &domain : bad_domains) { + check_is_email_address(userdata + '@' + domain, false); + check_is_email_address(userdata + domain, false); + } + for (auto &domain : good_domains) { + check_is_email_address(userdata + '@' + domain, false); + check_is_email_address(userdata + domain, false); + } + } + for (auto &userdata : good_userdatas) { + for (auto &domain : bad_domains) { + check_is_email_address(userdata + '@' + domain, false); + check_is_email_address(userdata + domain, false); + } + for (auto &domain : good_domains) { + check_is_email_address(userdata + '@' + domain, true); + check_is_email_address(userdata + domain, false); + } + } +} + +static void check_url(string str, std::vector<string> expected_urls, + std::vector<string> expected_email_addresses = {}) { + auto result_slice = find_urls(str); + std::vector<string> result_urls; + std::vector<string> result_email_addresses; + for (auto &it : result_slice) { + if (!it.second) { + result_urls.push_back(it.first.str()); + } else { + result_email_addresses.push_back(it.first.str()); + } + } + if (result_urls != expected_urls) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result_urls)) + << tag("expected", format::as_array(expected_urls)); + } + if (result_email_addresses != expected_email_addresses) { + LOG(FATAL) << tag("text", str) << tag("got", format::as_array(result_email_addresses)) + << tag("expected", format::as_array(expected_email_addresses)); + } +} + +TEST(MessageEntities, url) { + check_url("telegram.org", {"telegram.org"}); + check_url("(telegram.org)", {"telegram.org"}); + check_url("\ntelegram.org)", {"telegram.org"}); + check_url(" telegram.org)", {"telegram.org"}); + check_url(".telegram.org)", {}); + check_url("()telegram.org/?q=()", {"telegram.org/?q=()"}); + check_url("\"telegram.org\"", {"telegram.org"}); + check_url(" telegram. org. www. com... telegram.org... ...google.com...", {"telegram.org"}); + check_url(" telegram.org ", {"telegram.org"}); + check_url("Такой сайт: http://www.google.com или такой telegram.org ", {"http://www.google.com", "telegram.org"}); + check_url(" telegram.org. ", {"telegram.org"}); + check_url("http://google,.com", {}); + check_url("http://telegram.org/?asd=123#123.", {"http://telegram.org/?asd=123#123"}); + check_url("[http://google.com](test)", {"http://google.com"}); + check_url("", {}); + check_url(".", {}); + check_url("http://@google.com", {}); + check_url("http://@goog.com", {}); // TODO: server fix + check_url("http://@@google.com", {"http://@@google.com"}); + check_url("http://a@google.com", {"http://a@google.com"}); + check_url("http://test@google.com", {"http://test@google.com"}); + check_url("google.com:᪉᪉᪉᪉᪉", {"google.com"}); + check_url("https://telegram.org", {"https://telegram.org"}); + check_url("http://telegram.org", {"http://telegram.org"}); + check_url("ftp://telegram.org", {"ftp://telegram.org"}); + check_url("ftps://telegram.org", {}); + check_url("sftp://telegram.org", {"sftp://telegram.org"}); + check_url("hTtPs://telegram.org", {"hTtPs://telegram.org"}); + check_url("HTTP://telegram.org", {"HTTP://telegram.org"}); + check_url("аHTTP://telegram.org", {"HTTP://telegram.org"}); + check_url("sHTTP://telegram.org", {}); + check_url("://telegram.org", {}); + check_url("google.com:᪀᪀", {"google.com"}); + check_url( + "http://" + "abcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkab" + "cdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcd" + "efghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdef" + "ghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefgh" + "ijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghij" + "kabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijka" + "bcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabc" + "defghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijkabcdefghijk.com", + {}); + check_url("http:// .com", {}); + check_url("URL: .com", {}); + check_url("URL: .com", {}); + check_url(".com", {}); + check_url("http:// .", {}); + check_url("http://.", {}); + check_url("http://.com", {}); + check_url("http:// .", {}); + check_url(",ahttp://google.com", {"http://google.com"}); + check_url(".ahttp://google.com", {}); + check_url("http://1.0", {}); + check_url("http://a.0", {}); + check_url("http://a.a", {}); + check_url("google.com:1#ab c", {"google.com:1#ab"}); + check_url("google.com:1#", {"google.com:1#"}); + check_url("google.com:1#1", {"google.com:1#1"}); + check_url("google.com:00000001/abs", {"google.com:00000001/abs"}); + check_url("google.com:000000065535/abs", {"google.com:000000065535/abs"}); + check_url("google.com:000000065536/abs", {"google.com"}); + check_url("google.com:000000080/abs", {"google.com:000000080/abs"}); + check_url("google.com:0000000/abs", {"google.com"}); + check_url("google.com:0/abs", {"google.com"}); + check_url("google.com:/abs", {"google.com"}); + check_url("google.com:65535", {"google.com:65535"}); + check_url("google.com:65536", {"google.com"}); + check_url("google.com:99999", {"google.com"}); + check_url("google.com:100000", {"google.com"}); + check_url("127.001", {}); + check_url("127.0.0.1", {"127.0.0.1"}); + check_url("127.0.0.01", {}); + check_url("127.0.0.256", {}); + check_url("127.0.0.300", {}); + check_url("127.0.0.1000", {}); + check_url("127.0.0.260", {}); + check_url("1.0", {}); + check_url("www.🤙.tk", {"www.🤙.tk"}); + check_url("a.ab", {}); + check_url("test.abd", {}); + check_url("ТеСт.Москва", {}); + check_url("ТеСт.МоСкВΑ", {}); + check_url("ТеСт.МоСкВа", {"ТеСт.МоСкВа"}); + check_url("ТеСт.МоСкВач", {}); + check_url("http://ÀТеСт.МоСкВач", {"http://ÀТеСт.МоСкВач"}); + check_url("ÀÁ.com. ÀÁ.com.", {"ÀÁ.com", "ÀÁ.com"}); + check_url("ÀÁ.com,ÀÁ.com.", {"ÀÁ.com", "ÀÁ.com"}); + check_url("teiegram.org", {}); + check_url("http://test.google.com/?q=abc()}[]def", {"http://test.google.com/?q=abc()"}); + check_url("http://test.google.com/?q=abc([{)]}def", {"http://test.google.com/?q=abc([{)]}def"}); + check_url("http://test.google.com/?q=abc(){}]def", {"http://test.google.com/?q=abc(){}"}); + check_url("http://test.google.com/?q=abc){}[]def", {"http://test.google.com/?q=abc"}); + check_url("http://test.google.com/?q=abc(){}[]def", {"http://test.google.com/?q=abc(){}[]def"}); + check_url("http://test-.google.com", {}); + check_url("http://test_.google.com", {"http://test_.google.com"}); + check_url("http://google_.com", {}); + check_url("http://google._com_", {}); + check_url("http://[2001:4860:0:2001::68]/", {}); // TODO + check_url("test.abd", {}); + check_url("/.b/..a @.....@/. a.ba", {"a.ba"}); + check_url("bbbbbbbbbbbbbb.@.@", {}); + check_url("http://google.com/", {"http://google.com/"}); + check_url("http://google.com?", {"http://google.com"}); + check_url("http://google.com#", {"http://google.com#"}); + check_url("http://google.com/?", {"http://google.com/"}); + check_url("https://www.google.com/ab,", {"https://www.google.com/ab"}); + check_url("@.", {}); + check_url( + "a.b.google.com dfsknnfs gsdfgsg http://códuia.de/ dffdg,\" 12)(cpia.de/())(\" http://гришка.рф/ sdufhdf " + "http://xn--80afpi2a3c.xn--p1ai/ I have a good time.Thanks, guys!\n\n(hdfughidufhgdis)go#ogle.com гришка.рф " + "hsighsdf gi почта.рф\n\n✪df.ws/123 " + "xn--80afpi2a3c.xn--p1ai\n\nhttp://foo.com/blah_blah\nhttp://foo.com/blah_blah/\n(Something like " + "http://foo.com/blah_blah)\nhttp://foo.com/blah_blah_(wikipedi8989a_Вася)\n(Something like " + "http://foo.com/blah_blah_(Стакан_007))\nhttp://foo.com/blah_blah.\nhttp://foo.com/blah_blah/.\n<http://foo.com/" + "blah_blah>\n<http://fo@@@@@@@@@^%#*@^&@$#*@#%^*&!^#o.com/blah_blah/>\nhttp://foo.com/blah_blah,\nhttp://" + "www.example.com/wpstyle/?p=364.\nhttp://✪df.ws/123\nrdar://1234\nrdar:/1234\nhttp://" + "userid:password@example.com:8080\nhttp://userid@example.com\nhttp://userid@example.com:8080\nhttp://" + "userid:password@example.com\nhttp://example.com:8080 " + "x-yojimbo-item://6303E4C1-xxxx-45A6-AB9D-3A908F59AE0E\nmessage://" + "%3c330e7f8409726r6a4ba78dkf1fd71420c1bf6ff@mail.gmail.com%3e\nhttp://➡️.ws/䨹\nwww.➡️.ws/" + "䨹\n<tag>http://example.com</tag>\nJust a www.example.com " + "link.\n\n➡️.ws/" + "䨹\n\nabcdefghijklmnopqrstuvwxyz0123456789qwe_sdfsdf.aweawe-sdfs.com\nwww.🤙.tk:1\ngoogle.com:" + "᪉᪉᪉᪉\ngoogle." + "com:᪀᪀\nhttp:// .com\nURL: .com\nURL: " + ".com\n\ngoogle.com?qwe\ngoogle.com#qwe\ngoogle.com/?\ngoogle.com/#\ngoogle.com?\ngoogle.com#\n", + {"a.b.google.com", + "http://códuia.de/", + "cpia.de/()", + "http://гришка.рф/", + "http://xn--80afpi2a3c.xn--p1ai/", + "гришка.рф", + "почта.рф", + "✪df.ws/123", + "xn--80afpi2a3c.xn--p1ai", + "http://foo.com/blah_blah", + "http://foo.com/blah_blah/", + "http://foo.com/blah_blah", + "http://foo.com/blah_blah_(wikipedi8989a_Вася)", + "http://foo.com/blah_blah_(Стакан_007)", + "http://foo.com/blah_blah", + "http://foo.com/blah_blah/", + "http://foo.com/blah_blah", + "http://foo.com/blah_blah", + "http://www.example.com/wpstyle/?p=364", + "http://✪df.ws/123", + "http://userid:password@example.com:8080", + "http://userid@example.com", + "http://userid@example.com:8080", + "http://userid:password@example.com", + "http://example.com:8080", + "http://➡️.ws/䨹", + "www.➡️.ws/䨹", + "http://example.com", + "www.example.com", + "➡️.ws/䨹", + "abcdefghijklmnopqrstuvwxyz0123456789qwe_sdfsdf.aweawe-sdfs.com", + "www.🤙.tk:1", + "google.com", + "google.com", + "google.com?qwe", + "google.com#qwe", + "google.com/", + "google.com/#", + "google.com", + "google.com#"}); + check_url("https://t.…", {}); + check_url("('http://telegram.org/a-b/?br=ie&lang=en',)", {"http://telegram.org/a-b/?br=ie&lang=en"}); + check_url("https://ai.telegram.org/bot%20bot/test-...", {"https://ai.telegram.org/bot%20bot/test-"}); + check_url("a.bc@c.com", {}, {"a.bc@c.com"}); + check_url("https://a.bc@c.com", {"https://a.bc@c.com"}); + check_url("https://a.de[bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de/bc@c.com", {"https://a.de/bc@c.com"}); + check_url("https://a.de]bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de{bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de}bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de(bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de)bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de\\bc@c.com", {"https://a.de\\bc@c.com"}); + check_url("https://a.de'bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.de`bc@c.com", {"https://a.de"}, {"bc@c.com"}); + check_url("https://a.bc:de.fg@c.com", {"https://a.bc:de.fg@c.com"}); + check_url("https://a:h.bc:de.fg@c.com", {"https://a:h.bc:de.fg@c.com"}); + check_url("https://abc@c.com", {"https://abc@c.com"}); + check_url("https://de[bc@c.com", {}, {"bc@c.com"}); + check_url("https://de/bc@c.com", {}); + check_url("https://de]bc@c.com", {}, {"bc@c.com"}); + check_url("https://de{bc@c.com", {}, {"bc@c.com"}); + check_url("https://de}bc@c.com", {}, {"bc@c.com"}); + check_url("https://de(bc@c.com", {}, {"bc@c.com"}); + check_url("https://de)bc@c.com", {}, {"bc@c.com"}); + check_url("https://de\\bc@c.com", {"https://de\\bc@c.com"}); + check_url("https://de'bc@c.com", {}, {"bc@c.com"}); + check_url("https://de`bc@c.com", {}, {"bc@c.com"}); + check_url("https://bc:defg@c.com", {"https://bc:defg@c.com"}); + check_url("https://a:hbc:defg@c.com", {"https://a:hbc:defg@c.com"}); + check_url("https://a.bc@test.com:cd.com", {"https://a.bc@test.com", "cd.com"}); + check_url("telegram.Org", {}); + check_url("telegram.ORG", {"telegram.ORG"}); + check_url("a.b.c.com.a.b.c", {}); + check_url("File '/usr/views.py'", {}); // TODO server fix + check_url("@views.py'", {}); // TODO server fix + check_url("#views.py'", {}); // TODO server fix + check_url("/views.py'", {}); // TODO server fix + check_url(".views.py", {}); + check_url("'views.py'", {"views.py"}); + check_url("bug.http://test.com/test/+#", {}); // TODO {"http://test.com/test/+#"} + check_url("//AB.C.D.E.F.GH//", {}); + check_url("<http://www.ics.uci.edu/pub/ietf/uri/historical.html#WARNING>", + {"http://www.ics.uci.edu/pub/ietf/uri/historical.html#WARNING"}); + check_url("Look :test@example.com", {":test@example.com"}, {}); // TODO {}, {"test@example.com"} + check_url("http://test.com#a", {"http://test.com#a"}); + check_url("http://test.com#", {"http://test.com#"}); + check_url("http://test.com?#", {"http://test.com?#"}); + check_url("http://test.com/?#", {"http://test.com/?#"}); + check_url("https://t.me/abcdef…", {"https://t.me/abcdef"}); + check_url("https://t.me…", {"https://t.me"}); + check_url("https://t.m…", {}); + check_url("https://t.…", {}); + check_url("https://t…", {}); + check_url("👉http://ab.com/cdefgh-1IJ", {"http://ab.com/cdefgh-1IJ"}); + check_url("...👉http://ab.com/cdefgh-1IJ", {}); // TODO +} diff --git a/libs/tdlib/td/test/mtproto.cpp b/libs/tdlib/td/test/mtproto.cpp new file mode 100644 index 0000000000..7702a1e37b --- /dev/null +++ b/libs/tdlib/td/test/mtproto.cpp @@ -0,0 +1,347 @@ +// +// 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/actor/actor.h" +#include "td/actor/PromiseFuture.h" + +#include "td/mtproto/crypto.h" +#include "td/mtproto/Handshake.h" +#include "td/mtproto/HandshakeActor.h" +#include "td/mtproto/HandshakeConnection.h" +#include "td/mtproto/PingConnection.h" +#include "td/mtproto/RawConnection.h" + +#include "td/net/Socks5.h" + +#include "td/telegram/ConfigManager.h" +#include "td/telegram/net/PublicRsaKeyShared.h" + +#include "td/utils/logging.h" +#include "td/utils/port/IPAddress.h" +#include "td/utils/port/SocketFd.h" +#include "td/utils/Status.h" + +REGISTER_TESTS(mtproto); + +using namespace td; +using namespace mtproto; + +#if !TD_WINDOWS && !TD_EMSCRIPTEN // TODO +TEST(Mtproto, config) { + ConcurrentScheduler sched; + int threads_n = 0; + sched.init(threads_n); + + int cnt = 3; + { + auto guard = sched.get_current_guard(); + get_simple_config_azure(PromiseCreator::lambda([&](Result<SimpleConfig> r_simple_config) { + LOG(ERROR) << to_string(r_simple_config.ok()); + if (--cnt == 0) { + Scheduler::instance()->finish(); + } + })) + .release(); + + get_simple_config_google_app(PromiseCreator::lambda([&](Result<SimpleConfig> r_simple_config) { + LOG(ERROR) << to_string(r_simple_config.ok()); + if (--cnt == 0) { + Scheduler::instance()->finish(); + } + })) + .release(); + + get_simple_config_google_dns(PromiseCreator::lambda([&](Result<SimpleConfig> r_simple_config) { + LOG(ERROR) << to_string(r_simple_config.ok()); + if (--cnt == 0) { + Scheduler::instance()->finish(); + } + })) + .release(); + } + sched.start(); + while (sched.run_main(10)) { + // empty; + } + sched.finish(); +} +#endif + +TEST(Mtproto, encrypted_config) { + string data = + " LQ2 \b\n\tru6xVXpHHckW4eQWK0X3uThupVOor5sXT8t298IjDksYeUseQTOIrnUqiQj7o" + "+ZgPfhnfe+lfcQA+naax9akgllimjlJtL5riTf3O7iqZSnJ614qmCucxqqVTbXk/" + "hY2KaJTtsMqk7cShJjM3aQ4DD40h2InTaG7uyVO2q7K0GMUTeY3AM0Rt1lUjKHLD" + "g4RwjTzZaG8TwfgL/mZ7jsvgTTTATPWKUo7SmxQ9Hsj+07NMGqr6JKZS6aiU1Knz" + "VGCZ3OJEyRYocktN4HjaLpkguilaHWlVM2UNFUd5a+ajfLIiiKlH0FRC3XZ12CND" + "Y+NBjv0I57N2O4fBfswTlA== "; + auto config = decode_config(data).move_as_ok(); +} + +class TestPingActor : public Actor { + public: + TestPingActor(IPAddress ip_address, Status *result) : ip_address_(ip_address), result_(result) { + } + + private: + IPAddress ip_address_; + std::unique_ptr<mtproto::PingConnection> ping_connection_; + Status *result_; + + void start_up() override { + ping_connection_ = std::make_unique<mtproto::PingConnection>(std::make_unique<mtproto::RawConnection>( + SocketFd::open(ip_address_).move_as_ok(), mtproto::TransportType::Tcp, nullptr)); + + ping_connection_->get_pollable().set_observer(this); + subscribe(ping_connection_->get_pollable()); + set_timeout_in(10); + yield(); + } + void tear_down() override { + unsubscribe_before_close(ping_connection_->get_pollable()); + ping_connection_->close(); + Scheduler::instance()->finish(); + } + + void loop() override { + auto status = ping_connection_->flush(); + if (status.is_error()) { + *result_ = std::move(status); + return stop(); + } + if (ping_connection_->was_pong()) { + LOG(ERROR) << "GOT PONG"; + return stop(); + } + } + + void timeout_expired() override { + *result_ = Status::Error("Timeout expired"); + stop(); + } +}; + +static IPAddress get_default_ip_address() { + IPAddress ip_address; + ip_address.init_ipv4_port("149.154.167.40", 80).ensure(); + return ip_address; +} + +class Mtproto_ping : public td::Test { + public: + using Test::Test; + bool step() final { + if (!is_inited_) { + sched_.init(0); + sched_.create_actor_unsafe<TestPingActor>(0, "Pinger", get_default_ip_address(), &result_).release(); + sched_.start(); + is_inited_ = true; + } + + bool ret = sched_.run_main(10); + if (ret) { + return true; + } + sched_.finish(); + if (result_.is_error()) { + LOG(ERROR) << result_; + } + ASSERT_TRUE(result_.is_ok()); + return false; + } + + private: + bool is_inited_ = false; + ConcurrentScheduler sched_; + Status result_; +}; +Mtproto_ping mtproto_ping("Mtproto_ping"); + +class Context : public AuthKeyHandshakeContext { + public: + DhCallback *get_dh_callback() override { + return nullptr; + } + PublicRsaKeyInterface *get_public_rsa_key_interface() override { + return &public_rsa_key; + } + + private: + PublicRsaKeyShared public_rsa_key{DcId::empty()}; +}; + +class HandshakeTestActor : public Actor { + public: + explicit HandshakeTestActor(Status *result) : result_(result) { + } + + private: + Status *result_; + bool wait_for_raw_connection_ = false; + std::unique_ptr<RawConnection> raw_connection_; + bool wait_for_handshake_ = false; + std::unique_ptr<AuthKeyHandshake> handshake_; + Status status_; + bool wait_for_result_ = false; + + void tear_down() override { + if (raw_connection_) { + raw_connection_->close(); + } + finish(Status::Error("Interrupted")); + } + void loop() override { + if (!wait_for_raw_connection_ && !raw_connection_) { + raw_connection_ = std::make_unique<mtproto::RawConnection>(SocketFd::open(get_default_ip_address()).move_as_ok(), + mtproto::TransportType::Tcp, nullptr); + } + if (!wait_for_handshake_ && !handshake_) { + handshake_ = std::make_unique<AuthKeyHandshake>(0); + } + if (raw_connection_ && handshake_) { + if (wait_for_result_) { + wait_for_result_ = false; + if (status_.is_error()) { + finish(std::move(status_)); + return stop(); + } + if (!handshake_->is_ready_for_finish()) { + finish(Status::Error("Key is not ready..")); + return stop(); + } + finish(Status::OK()); + return stop(); + } + + wait_for_result_ = true; + create_actor<HandshakeActor>( + "HandshakeActor", std::move(handshake_), std::move(raw_connection_), std::make_unique<Context>(), 10.0, + PromiseCreator::lambda([self = actor_id(this)](Result<std::unique_ptr<RawConnection>> raw_connection) { + send_closure(self, &HandshakeTestActor::got_connection, std::move(raw_connection), 1); + }), + PromiseCreator::lambda([self = actor_id(this)](Result<std::unique_ptr<AuthKeyHandshake>> handshake) { + send_closure(self, &HandshakeTestActor::got_handshake, std::move(handshake), 1); + })) + .release(); + wait_for_raw_connection_ = true; + wait_for_handshake_ = true; + } + } + + void got_connection(Result<std::unique_ptr<RawConnection>> r_raw_connection, int32 dummy) { + CHECK(wait_for_raw_connection_); + wait_for_raw_connection_ = false; + if (r_raw_connection.is_ok()) { + raw_connection_ = r_raw_connection.move_as_ok(); + status_ = Status::OK(); + } else { + status_ = r_raw_connection.move_as_error(); + } + // TODO: save error + loop(); + } + + void got_handshake(Result<std::unique_ptr<AuthKeyHandshake>> r_handshake, int32 dummy) { + CHECK(wait_for_handshake_); + wait_for_handshake_ = false; + CHECK(r_handshake.is_ok()); + handshake_ = r_handshake.move_as_ok(); + loop(); + } + + void finish(Status status) { + if (!result_) { + return; + } + *result_ = std::move(status); + result_ = nullptr; + Scheduler::instance()->finish(); + } +}; + +class Mtproto_handshake : public td::Test { + public: + using Test::Test; + bool step() final { + if (!is_inited_) { + sched_.init(0); + sched_.create_actor_unsafe<HandshakeTestActor>(0, "HandshakeTestActor", &result_).release(); + sched_.start(); + is_inited_ = true; + } + + bool ret = sched_.run_main(10); + if (ret) { + return true; + } + sched_.finish(); + if (result_.is_error()) { + LOG(ERROR) << result_; + } + ASSERT_TRUE(result_.is_ok()); + return false; + } + + private: + bool is_inited_ = false; + ConcurrentScheduler sched_; + Status result_; +}; +Mtproto_handshake mtproto_handshake("Mtproto_handshake"); + +class Socks5TestActor : public Actor { + public: + void start_up() override { + auto promise = PromiseCreator::lambda([actor_id = actor_id(this)](Result<SocketFd> res) { + send_closure(actor_id, &Socks5TestActor::on_result, std::move(res), false); + }); + + class Callback : public Socks5::Callback { + public: + explicit Callback(Promise<SocketFd> promise) : promise_(std::move(promise)) { + } + void set_result(Result<SocketFd> result) override { + promise_.set_result(std::move(result)); + } + void on_connected() override { + } + + private: + Promise<SocketFd> promise_; + }; + + IPAddress socks5_ip; + socks5_ip.init_ipv4_port("131.191.89.104", 43077).ensure(); + IPAddress mtproto_ip = get_default_ip_address(); + + auto r_socket = SocketFd::open(socks5_ip); + create_actor<Socks5>("socks5", r_socket.move_as_ok(), mtproto_ip, "", "", + std::make_unique<Callback>(std::move(promise)), actor_shared()) + .release(); + } + + private: + void on_result(Result<SocketFd> res, bool dummy) { + res.ensure(); + Scheduler::instance()->finish(); + } +}; + +TEST(Mtproto, socks5) { + return; + ConcurrentScheduler sched; + int threads_n = 0; + sched.init(threads_n); + + sched.create_actor_unsafe<Socks5TestActor>(0, "Socks5TestActor").release(); + sched.start(); + while (sched.run_main(10)) { + // empty; + } + sched.finish(); +} diff --git a/libs/tdlib/td/test/secret.cpp b/libs/tdlib/td/test/secret.cpp new file mode 100644 index 0000000000..2abed9787b --- /dev/null +++ b/libs/tdlib/td/test/secret.cpp @@ -0,0 +1,1056 @@ +// +// 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/actor/PromiseFuture.h" + +#include "td/db/binlog/detail/BinlogEventsProcessor.h" + +#include "td/mtproto/crypto.h" +#include "td/mtproto/utils.h" + +#include "td/telegram/Global.h" +#include "td/telegram/MessageId.h" +#include "td/telegram/SecretChatActor.h" +#include "td/telegram/SecretChatId.h" + +#include "td/telegram/secret_api.h" +#include "td/telegram/telegram_api.h" + +#include "td/tl/tl_object_parse.h" +#include "td/tl/tl_object_store.h" + +#include "td/utils/base64.h" +#include "td/utils/buffer.h" +#include "td/utils/crypto.h" +#include "td/utils/format.h" +#include "td/utils/Gzip.h" +#include "td/utils/logging.h" +#include "td/utils/misc.h" +#include "td/utils/Random.h" +#include "td/utils/Slice.h" +#include "td/utils/Status.h" +#include "td/utils/tests.h" +#include "td/utils/tl_helpers.h" +#include "td/utils/tl_parsers.h" +#include "td/utils/tl_storers.h" + +#include <cstdio> +#include <ctime> +#include <limits> +#include <map> +#include <memory> + +REGISTER_TESTS(secret); + +namespace my_api { + +using namespace td; + +//messages_getDhConfig +class messages_getDhConfig { + public: + int32 version_; + int32 random_length_; + + messages_getDhConfig() = default; + + messages_getDhConfig(int32 version_, int32 random_length_); + + static const int32 ID = 651135312; + + explicit messages_getDhConfig(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : version_(TlFetchInt::parse(p)) + , random_length_(TlFetchInt::parse(p)) +#undef FAIL + { + } +}; + +//InputUser +class InputUser { + public: + static tl_object_ptr<InputUser> fetch(TlBufferParser &p); +}; + +class inputUser final : public InputUser { + public: + int32 user_id_; + int64 access_hash_; + + static const int32 ID = -668391402; + inputUser() = default; + + explicit inputUser(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : user_id_(TlFetchInt::parse(p)) + , access_hash_(TlFetchLong::parse(p)) +#undef FAIL + { + } +}; +tl_object_ptr<InputUser> InputUser::fetch(TlBufferParser &p) { +#define FAIL(error) \ + p.set_error(error); \ + return nullptr; + int constructor = p.fetch_int(); + switch (constructor) { + case inputUser::ID: + return make_tl_object<inputUser>(p); + default: + FAIL(PSTRING() << "Unknown constructor found " << format::as_hex(constructor)); + } +#undef FAIL +} + +class messages_requestEncryption final { + public: + tl_object_ptr<InputUser> user_id_; + int32 random_id_; + BufferSlice g_a_; + + static const int32 ID = -162681021; + messages_requestEncryption(); + + explicit messages_requestEncryption(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : user_id_(TlFetchObject<InputUser>::parse(p)) + , random_id_(TlFetchInt::parse(p)) + , g_a_(TlFetchBytes<BufferSlice>::parse(p)) +#undef FAIL + { + } +}; + +class inputEncryptedChat final { + public: + int32 chat_id_; + int64 access_hash_; + + inputEncryptedChat() = default; + + static const int32 ID = -247351839; + explicit inputEncryptedChat(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : chat_id_(TlFetchInt::parse(p)) + , access_hash_(TlFetchLong::parse(p)) +#undef FAIL + { + } + static tl_object_ptr<inputEncryptedChat> fetch(TlBufferParser &p) { + return make_tl_object<inputEncryptedChat>(p); + } +}; + +class messages_acceptEncryption final { + public: + tl_object_ptr<inputEncryptedChat> peer_; + BufferSlice g_b_; + int64 key_fingerprint_; + + messages_acceptEncryption() = default; + + static const int32 ID = 1035731989; + + explicit messages_acceptEncryption(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : peer_(TlFetchBoxed<TlFetchObject<inputEncryptedChat>, -247351839>::parse(p)) + , g_b_(TlFetchBytes<BufferSlice>::parse(p)) + , key_fingerprint_(TlFetchLong::parse(p)) +#undef FAIL + { + } +}; + +class messages_sendEncryptedService final { + public: + tl_object_ptr<inputEncryptedChat> peer_; + int64 random_id_; + BufferSlice data_; + + messages_sendEncryptedService() = default; + static const int32 ID = 852769188; + explicit messages_sendEncryptedService(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : peer_(TlFetchBoxed<TlFetchObject<inputEncryptedChat>, -247351839>::parse(p)) + , random_id_(TlFetchLong::parse(p)) + , data_(TlFetchBytes<BufferSlice>::parse(p)) +#undef FAIL + { + } +}; + +class messages_sendEncrypted final { + public: + tl_object_ptr<inputEncryptedChat> peer_; + int64 random_id_; + BufferSlice data_; + + messages_sendEncrypted() = default; + static const int32 ID = -1451792525; + + explicit messages_sendEncrypted(TlBufferParser &p) +#define FAIL(error) p.set_error(error) + : peer_(TlFetchBoxed<TlFetchObject<inputEncryptedChat>, -247351839>::parse(p)) + , random_id_(TlFetchLong::parse(p)) + , data_(TlFetchBytes<BufferSlice>::parse(p)) +#undef FAIL + { + } +}; + +template <class F> +static void downcast_call(TlBufferParser &p, F &&f) { + auto id = p.fetch_int(); + switch (id) { + case messages_getDhConfig::ID: + return f(*make_tl_object<messages_getDhConfig>(p)); + case messages_requestEncryption::ID: + return f(*make_tl_object<messages_requestEncryption>(p)); + case messages_acceptEncryption::ID: + return f(*make_tl_object<messages_acceptEncryption>(p)); + case messages_sendEncrypted::ID: + return f(*make_tl_object<messages_sendEncrypted>(p)); + case messages_sendEncryptedService::ID: + return f(*make_tl_object<messages_sendEncryptedService>(p)); + default: + CHECK(0) << id; + UNREACHABLE(); + } +} + +class messages_dhConfig final { + public: + int32 g_; + BufferSlice p_; + int32 version_; + BufferSlice random_; + + messages_dhConfig() = default; + + messages_dhConfig(int32 g_, BufferSlice &&p_, int32 version_, BufferSlice &&random_) + : g_(g_), p_(std::move(p_)), version_(version_), random_(std::move(random_)) { + } + + static const int32 ID = 740433629; + int32 get_id() const { + return ID; + } + + void store(TlStorerCalcLength &s) const { + (void)sizeof(s); + TlStoreBinary::store(g_, s); + TlStoreString::store(p_, s); + TlStoreBinary::store(version_, s); + TlStoreString::store(random_, s); + } + void store(TlStorerUnsafe &s) const { + (void)sizeof(s); + TlStoreBinary::store(g_, s); + TlStoreString::store(p_, s); + TlStoreBinary::store(version_, s); + TlStoreString::store(random_, s); + } +}; + +class encryptedChat final { + public: + int32 id_; + int64 access_hash_; + int32 date_; + int32 admin_id_; + int32 participant_id_; + BufferSlice g_a_or_b_; + int64 key_fingerprint_; + + encryptedChat() = default; + + encryptedChat(int32 id_, int64 access_hash_, int32 date_, int32 admin_id_, int32 participant_id_, + BufferSlice &&g_a_or_b_, int64 key_fingerprint_) + : id_(id_) + , access_hash_(access_hash_) + , date_(date_) + , admin_id_(admin_id_) + , participant_id_(participant_id_) + , g_a_or_b_(std::move(g_a_or_b_)) + , key_fingerprint_(key_fingerprint_) { + } + + static const int32 ID = -94974410; + int32 get_id() const { + return ID; + } + + void store(TlStorerCalcLength &s) const { + (void)sizeof(s); + TlStoreBinary::store(id_, s); + TlStoreBinary::store(access_hash_, s); + TlStoreBinary::store(date_, s); + TlStoreBinary::store(admin_id_, s); + TlStoreBinary::store(participant_id_, s); + TlStoreString::store(g_a_or_b_, s); + TlStoreBinary::store(key_fingerprint_, s); + } + + void store(TlStorerUnsafe &s) const { + (void)sizeof(s); + TlStoreBinary::store(id_, s); + TlStoreBinary::store(access_hash_, s); + TlStoreBinary::store(date_, s); + TlStoreBinary::store(admin_id_, s); + TlStoreBinary::store(participant_id_, s); + TlStoreString::store(g_a_or_b_, s); + TlStoreBinary::store(key_fingerprint_, s); + } +}; + +class messages_sentEncryptedMessage final { + public: + int32 date_; + + messages_sentEncryptedMessage() = default; + + explicit messages_sentEncryptedMessage(int32 date_) : date_(date_) { + } + + static const int32 ID = 1443858741; + int32 get_id() const { + return ID; + } + + void store(TlStorerCalcLength &s) const { + (void)sizeof(s); + TlStoreBinary::store(date_, s); + } + + void store(TlStorerUnsafe &s) const { + (void)sizeof(s); + TlStoreBinary::store(date_, s); + } +}; + +} // namespace my_api + +namespace td { +static int32 g = 3; +static string prime_base64 = + "xxyuucaxyQSObFIvcPE_c5gNQCOOPiHBSTTQN1Y9kw9IGYoKp8FAWCKUk9IlMPTb-jNvbgrJJROVQ67UTM58NyD9UfaUWHBaxozU_mtrE6vcl0ZRKW" + "kyhFTxj6-MWV9kJHf-lrsqlB1bzR1KyMxJiAcI-ps3jjxPOpBgvuZ8-aSkppWBEFGQfhYnU7VrD2tBDbp02KhLKhSzFE4O8ShHVP0X7ZUNWWW0ud1G" + "WC2xF40WnGvEZbDW_5yjko_vW5rk5Bj8Feg-vqD4f6n_Xu1wBQ3tKEn0e_lZ2VaFDOkphR8NgRX2NbEF7i5OFdBLJFS_b0-t8DSxBAMRnNjjuS_MW" + "w"; + +class FakeDhCallback : public DhCallback { + public: + int is_good_prime(Slice prime_str) const override { + auto it = cache.find(prime_str.str()); + if (it == cache.end()) { + return -1; + } + return it->second; + } + void add_good_prime(Slice prime_str) const override { + cache[prime_str.str()] = 1; + } + void add_bad_prime(Slice prime_str) const override { + cache[prime_str.str()] = 0; + } + mutable std::map<string, int> cache; +}; + +class FakeBinlog + : public BinlogInterface + , public Actor { + public: + FakeBinlog() { + register_actor("FakeBinlog", this).release(); + } + void force_sync(Promise<> promise) override { + if (pending_events_.empty()) { + pending_events_.emplace_back(); + } + pending_events_.back().promises_.push_back(std::move(promise)); + pending_events_.back().sync_flag = true; + request_sync(); + } + void request_sync() { + if (!has_request_sync) { + has_request_sync = true; + if (Random::fast(0, 4) == 0) { + set_timeout_in(Random::fast(0, 99) / 100.0 * 0.005 + 0.001); + } else { + yield(); + } + } + } + void force_flush() override { + } + + uint64 next_id() override { + auto res = last_id_; + last_id_++; + return res; + } + uint64 next_id(int32 shift) override { + auto res = last_id_; + last_id_ += shift; + return res; + } + template <class F> + void for_each(const F &f) { + events_processor_.for_each([&](auto &x) { + LOG(INFO) << "REPLAY: " << x.id_; + f(x); + }); + } + + void restart() { + has_request_sync = false; + cancel_timeout(); + for (auto &pending : pending_events_) { + auto &event = pending.event; + if (!event.empty()) { + // LOG(ERROR) << "FORGET EVENT: " << event.id_ << " " << event; + } + } + pending_events_.clear(); + } + + void change_key(DbKey key, Promise<> promise) override { + } + + protected: + void close_impl(Promise<> promise) override { + } + void close_and_destroy_impl(Promise<> promise) override { + } + void add_raw_event_impl(uint64 id, BufferSlice &&raw_event, Promise<> promise) override { + auto event = BinlogEvent(std::move(raw_event)); + LOG(INFO) << "ADD EVENT: " << event.id_ << " " << event; + pending_events_.emplace_back(); + pending_events_.back().event = std::move(event); + pending_events_.back().promises_.push_back(std::move(promise)); + } + void do_force_sync() { + if (pending_events_.empty()) { + return; + } + cancel_timeout(); + has_request_sync = false; + auto pos = static_cast<size_t>(Random::fast_uint64() % pending_events_.size()); + // pos = pending_events_.size() - 1; + std::vector<Promise<>> promises; + for (size_t i = 0; i <= pos; i++) { + auto &pending = pending_events_[i]; + auto event = std::move(pending.event); + if (!event.empty()) { + LOG(INFO) << "SAVE EVENT: " << event.id_ << " " << event; + events_processor_.add_event(std::move(event)); + } + append(promises, std::move(pending.promises_)); + } + pending_events_.erase(pending_events_.begin(), pending_events_.begin() + pos + 1); + for (auto &promise : promises) { + promise.set_value(Unit()); + } + + for (auto &event : pending_events_) { + if (event.sync_flag) { + request_sync(); + break; + } + } + } + void timeout_expired() override { + do_force_sync(); + } + void wakeup() override { + if (has_request_sync) { + do_force_sync(); + } + } + bool has_request_sync = false; + uint64 last_id_ = 1; + detail::BinlogEventsProcessor events_processor_; + + struct PendingEvent { + BinlogEvent event; + bool sync_flag = false; + std::vector<Promise<>> promises_; + }; + + std::vector<PendingEvent> pending_events_; +}; + +using FakeKeyValue = BinlogKeyValue<BinlogInterface>; +class OldFakeKeyValue : public KeyValueSyncInterface { + SeqNo set(string key, string value) override { + kv_[key] = value; + return 0; + } + + SeqNo erase(const string &key) override { + kv_.erase(key); + return 0; + } + + bool isset(const string &key) override { + return kv_.count(key) > 0; + } + + string get(const string &key) override { + auto it = kv_.find(key); + if (it != kv_.end()) { + return it->second; + } + return string(); + } + + private: + std::map<string, string> kv_; +}; + +class Master; +class FakeSecretChatContext : public SecretChatActor::Context { + public: + FakeSecretChatContext(std::shared_ptr<BinlogInterface> binlog, std::shared_ptr<KeyValueSyncInterface> key_value, + std::shared_ptr<bool> close_flag, ActorShared<Master> master) + : binlog_(std::move(binlog)) + , key_value_(std::move(key_value)) + , close_flag_(std::move(close_flag)) + , master_(std::move(master)) { + secret_chat_db_ = std::make_unique<SecretChatDb>(key_value_, 1); + net_query_creator_.stop_check(); // :( + } + DhCallback *dh_callback() override { + return &fake_dh_callback_; + } + NetQueryCreator &net_query_creator() override { + return net_query_creator_; + } + int32 unix_time() override { + return static_cast<int32>(std::time(nullptr)); + } + bool close_flag() override { + return *close_flag_; + } + BinlogInterface *binlog() override { + return binlog_.get(); + } + SecretChatDb *secret_chat_db() override { + return secret_chat_db_.get(); + } + std::shared_ptr<DhConfig> dh_config() override { + static auto config = [] { + DhConfig dh_config; + dh_config.version = 12; + dh_config.g = g; + dh_config.prime = base64url_decode(prime_base64).move_as_ok(); + return std::make_shared<DhConfig>(dh_config); + }(); + + return config; + } + void set_dh_config(std::shared_ptr<DhConfig> dh_config) override { + // empty + } + + bool get_config_option_boolean(const string &name) const override { + return false; + } + + // We don't want to expose the whole NetQueryDispatcher, MessagesManager and ContactsManager. + // So it is more clear which parts of MessagesManager is really used. And it is much easier to create tests. + void send_net_query(NetQueryPtr query, ActorShared<NetQueryCallback> callback, bool ordered) override; + + void on_update_secret_chat(int64 access_hash, UserId user_id, SecretChatState state, bool is_outbound, int32 ttl, + int32 date, string key_hash, int32 layer) override { + } + + void on_inbound_message(UserId user_id, MessageId message_id, int32 date, + tl_object_ptr<telegram_api::encryptedFile> file, + tl_object_ptr<secret_api::decryptedMessage> message, Promise<>) override; + + void on_send_message_error(int64 random_id, Status error, Promise<>) override; + void on_send_message_ack(int64 random_id) override; + void on_send_message_ok(int64 random_id, MessageId message_id, int32 date, + tl_object_ptr<telegram_api::EncryptedFile> file, Promise<>) override; + void on_delete_messages(std::vector<int64> random_id, Promise<>) override; + void on_flush_history(MessageId, Promise<>) override; + void on_read_message(int64, Promise<>) override; + + void on_screenshot_taken(UserId user_id, MessageId message_id, int32 date, int64 random_id, + Promise<> promise) override { + } + void on_set_ttl(UserId user_id, MessageId message_id, int32 date, int32 ttl, int64 random_id, + Promise<> promise) override { + } + + private: + FakeDhCallback fake_dh_callback_; + static NetQueryCreator net_query_creator_; + std::shared_ptr<BinlogInterface> binlog_; + std::shared_ptr<KeyValueSyncInterface> key_value_; + std::shared_ptr<bool> close_flag_; + ActorShared<Master> master_; + + std::shared_ptr<SecretChatDb> secret_chat_db_; +}; +NetQueryCreator FakeSecretChatContext::net_query_creator_; + +class Master : public Actor { + public: + explicit Master(Status *status) : status_(status) { + } + class SecretChatProxy : public Actor { + public: + SecretChatProxy(string name, ActorShared<Master> parent) : name_(std::move(name)) { + binlog_ = std::make_shared<FakeBinlog>(); + key_value_ = std::make_shared<FakeKeyValue>(); + key_value_->external_init_begin(LogEvent::HandlerType::BinlogPmcMagic); + key_value_->external_init_finish(binlog_); + close_flag_ = std::make_shared<bool>(false); + parent_ = parent.get(); + parent_token_ = parent.token(); + actor_ = create_actor<SecretChatActor>( + "SecretChat " + name_, 123, + std::make_unique<FakeSecretChatContext>(binlog_, key_value_, close_flag_, std::move(parent)), true); + on_binlog_replay_finish(); + } + + ActorOwn<SecretChatActor> actor_; + + void add_inbound_message(int32 chat_id, BufferSlice data, uint64 crc) { + CHECK(crc64(data.as_slice()) == crc); + auto event = std::make_unique<logevent::InboundSecretMessage>(); + event->qts = 0; + event->chat_id = chat_id; + event->date = 0; + event->encrypted_message = std::move(data); + event->qts_ack = PromiseCreator::lambda( + [actor_id = actor_id(this), chat_id, data = event->encrypted_message.copy(), crc](Result<> result) mutable { + if (result.is_ok()) { + LOG(INFO) << "FINISH add_inbound_message " << tag("crc", crc); + return; + } + LOG(INFO) << "RESEND add_inbound_message " << tag("crc", crc) << result.error(); + send_closure(actor_id, &SecretChatProxy::add_inbound_message, chat_id, std::move(data), crc); + }); + + add_event(Event::delayed_closure(&SecretChatActor::add_inbound_message, std::move(event))); + } + + void send_message(tl_object_ptr<secret_api::decryptedMessage> message) { + BufferSlice serialized_message(serialize(*message)); + auto resend_promise = PromiseCreator::lambda( + [actor_id = actor_id(this), serialized_message = std::move(serialized_message)](Result<> result) mutable { + TlBufferParser parser(&serialized_message); + auto message = secret_api::decryptedMessage::fetch(parser); + if (result.is_ok()) { + LOG(INFO) << "FINISH send_message " << tag("message", to_string(message)); + return; + } + LOG(INFO) << "RESEND send_message " << tag("message", to_string(message)) << result.error(); + CHECK(serialize(*message) == serialized_message.as_slice()); + send_closure(actor_id, &SecretChatProxy::send_message, std::move(message)); + }); + auto sync_promise = PromiseCreator::lambda([actor_id = actor_id(this), generation = this->binlog_generation_, + resend_promise = std::move(resend_promise)](Result<> result) mutable { + if (result.is_error()) { + resend_promise.set_error(result.move_as_error()); + return; + } + send_closure(actor_id, &SecretChatProxy::sync_binlog, generation, std::move(resend_promise)); + }); + + add_event( + Event::delayed_closure(&SecretChatActor::send_message, std::move(message), nullptr, std::move(sync_promise))); + } + int32 binlog_generation_ = 0; + void sync_binlog(int32 binlog_generation, Promise<> promise) { + if (binlog_generation != binlog_generation_) { + return promise.set_error(Status::Error("binlog generation mismatch")); + } + binlog_->force_sync(std::move(promise)); + } + void on_closed() { + LOG(INFO) << "CLOSED"; + ready_ = false; + *close_flag_ = false; + + key_value_ = std::make_shared<FakeKeyValue>(); + key_value_->external_init_begin(LogEvent::HandlerType::BinlogPmcMagic); + + std::vector<BinlogEvent> events; + binlog_generation_++; + binlog_->restart(); + binlog_->for_each([&](const BinlogEvent &event) { + if (event.type_ == LogEvent::HandlerType::BinlogPmcMagic) { + key_value_->external_init_handle(event); + } else { + events.push_back(event.clone()); + } + }); + + key_value_->external_init_finish(binlog_); + + actor_ = create_actor<SecretChatActor>( + "SecretChat " + name_, 123, + std::make_unique<FakeSecretChatContext>(binlog_, key_value_, close_flag_, + ActorShared<Master>(parent_, parent_token_)), + true); + + for (auto &event : events) { + CHECK(event.type_ == LogEvent::HandlerType::SecretChats); + auto r_message = logevent::SecretChatEvent::from_buffer_slice(event.data_as_buffer_slice()); + LOG_IF(FATAL, r_message.is_error()) << "Failed to deserialize event: " << r_message.error(); + auto message = r_message.move_as_ok(); + message->set_logevent_id(event.id_); + LOG(INFO) << "Process binlog event " << *message; + switch (message->get_type()) { + case logevent::SecretChatEvent::Type::InboundSecretMessage: + send_closure_later(actor_, &SecretChatActor::replay_inbound_message, + std::unique_ptr<logevent::InboundSecretMessage>( + static_cast<logevent::InboundSecretMessage *>(message.release()))); + break; + case logevent::SecretChatEvent::Type::OutboundSecretMessage: + send_closure_later(actor_, &SecretChatActor::replay_outbound_message, + std::unique_ptr<logevent::OutboundSecretMessage>( + static_cast<logevent::OutboundSecretMessage *>(message.release()))); + break; + default: + UNREACHABLE(); + } + }; + start_test(); + on_binlog_replay_finish(); + } + void on_binlog_replay_finish() { + ready_ = true; + LOG(INFO) << "on_binlog_replay_finish!"; + send_closure(actor_, &SecretChatActor::binlog_replay_finish); + for (auto &event : pending_events_) { + send_event(actor_, std::move(event)); + } + pending_events_.clear(); + } + void start_test() { + set_timeout_in(Random::fast(50, 99) * 0.3 / 50); + events_cnt_ = 0; + } + + private: + string name_; + + ActorId<Master> parent_; + uint64 parent_token_; + std::shared_ptr<FakeBinlog> binlog_; + std::shared_ptr<FakeKeyValue> key_value_; + std::shared_ptr<bool> close_flag_; + int events_cnt_ = 0; + + std::vector<Event> pending_events_; + bool ready_ = false; + + bool is_active() { + return !actor_.empty() && ready_; + } + void add_event(Event event) { + events_cnt_++; + if (is_active()) { + LOG(INFO) << "EMIT"; + send_event(actor_, std::move(event)); + } else { + LOG(INFO) << "DELAY"; + pending_events_.push_back(std::move(event)); + } + } + + int32 bad_cnt_ = 0; + void timeout_expired() override { + LOG(INFO) << "TIMEOUT EXPIRED"; + if (events_cnt_ < 4) { + bad_cnt_++; + CHECK(bad_cnt_ < 10); + } else { + bad_cnt_ = 0; + } + *close_flag_ = true; + actor_.reset(); + } + }; + + auto &get_by_id(uint64 id) { + if (id == 1) { + return alice_; + } else { + return bob_; + } + } + auto &from() { + return get_by_id(get_link_token()); + } + auto &to() { + return get_by_id(3 - get_link_token()); + } + void start_up() override { + set_context(std::make_shared<Global>()); + alice_ = create_actor<SecretChatProxy>("SecretChatProxy alice", "alice", actor_shared(this, 1)); + bob_ = create_actor<SecretChatProxy>("SecretChatProxy bob", "bob", actor_shared(this, 2)); + send_closure(alice_->get_actor_unsafe()->actor_, &SecretChatActor::create_chat, 2, 0, 123, + PromiseCreator::lambda([actor_id = actor_id(this)](Result<SecretChatId> res) { + send_closure(actor_id, &Master::got_secret_chat_id, std::move(res), 0); + })); + } + void got_secret_chat_id(Result<SecretChatId> res, int) { // second parameter is needed to workaround clang bug + CHECK(res.is_ok()); + auto id = res.move_as_ok(); + LOG(INFO) << "SecretChatId = " << id; + } + bool can_fail(NetQueryPtr &query) { + static int cnt = 20; + if (cnt > 0) { + cnt--; + return false; + } + if (query->tl_constructor() == telegram_api::messages_sendEncrypted::ID || + query->tl_constructor() == telegram_api::messages_sendEncryptedFile::ID) { + return true; + } + return false; + } + void send_net_query(NetQueryPtr query, ActorShared<NetQueryCallback> callback, bool ordered) { + if (can_fail(query) && Random::fast(0, 1) == 0) { + LOG(INFO) << "Fail query " << query; + auto resend_promise = + PromiseCreator::lambda([id = actor_shared(this, get_link_token()), callback_actor = callback.get(), + callback_token = callback.token()](Result<NetQueryPtr> r_net_query) mutable { + if (r_net_query.is_error()) { + id.release(); + return; + } + send_closure(std::move(id), &Master::send_net_query, r_net_query.move_as_ok(), + ActorShared<NetQueryCallback>(callback_actor, callback_token), true); + }); + query->set_error(Status::Error(429, "Test error")); + send_closure(std::move(callback), &NetQueryCallback::on_result_resendable, std::move(query), + std::move(resend_promise)); + return; + } else { + LOG(INFO) << "Do not fail " << query; + } + auto query_slice = query->query().clone(); + if (query->gzip_flag() == NetQuery::GzipFlag::On) { + query_slice = gzdecode(query_slice.as_slice()); + } + TlBufferParser parser(&query_slice); + //auto object = telegram_api::Function::fetch(parser); + //LOG(INFO) << query_slice.size(); + //parser.get_status().ensure(); + my_api::downcast_call(parser, [&](auto &object) { + this->process_net_query(std::move(object), std::move(query), std::move(callback)); + }); + } + template <class T> + void process_net_query(T &&object, NetQueryPtr query, ActorShared<NetQueryCallback> callback) { + LOG(FATAL) << "Unsupported query: " << to_string(object); + } + void process_net_query(my_api::messages_getDhConfig &&get_dh_config, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + //LOG(INFO) << "Got query " << to_string(get_dh_config); + my_api::messages_dhConfig config; + config.p_ = BufferSlice(base64url_decode(prime_base64).move_as_ok()); + config.g_ = g; + config.version_ = 12; + auto storer = TLObjectStorer<my_api::messages_dhConfig>(config); + BufferSlice answer(storer.size()); + storer.store(answer.as_slice().ubegin()); + net_query->set_ok(std::move(answer)); + send_closure(std::move(callback), &NetQueryCallback::on_result, std::move(net_query)); + } + void process_net_query(my_api::messages_requestEncryption &&request_encryption, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + CHECK(get_link_token() == 1); + send_closure(alice_->get_actor_unsafe()->actor_, &SecretChatActor::update_chat, + make_tl_object<telegram_api::encryptedChatWaiting>(123, 321, 0, 1, 2)); + send_closure( + bob_->get_actor_unsafe()->actor_, &SecretChatActor::update_chat, + make_tl_object<telegram_api::encryptedChatRequested>(123, 321, 0, 1, 2, request_encryption.g_a_.clone())); + net_query->clear(); + } + void process_net_query(my_api::messages_acceptEncryption &&request_encryption, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + CHECK(get_link_token() == 2); + send_closure(alice_->get_actor_unsafe()->actor_, &SecretChatActor::update_chat, + make_tl_object<telegram_api::encryptedChat>(123, 321, 0, 1, 2, request_encryption.g_b_.clone(), + request_encryption.key_fingerprint_)); + + my_api::encryptedChat encrypted_chat(123, 321, 0, 1, 2, BufferSlice(), request_encryption.key_fingerprint_); + auto storer = TLObjectStorer<my_api::encryptedChat>(encrypted_chat); + BufferSlice answer(storer.size()); + storer.store(answer.as_slice().ubegin()); + net_query->set_ok(std::move(answer)); + send_closure(std::move(callback), &NetQueryCallback::on_result, std::move(net_query)); + send_closure(alice_, &SecretChatProxy::start_test); + send_closure(bob_, &SecretChatProxy::start_test); + send_ping(1, 5000); + set_timeout_in(1); + } + void timeout_expired() override { + send_message(1, "oppa"); + send_message(2, "appo"); + set_timeout_in(1); + } + void send_ping(int id, int cnt) { + if (cnt % 200 == 0) { + LOG(ERROR) << "send ping " << tag("id", id) << tag("cnt", cnt); + } else { + LOG(INFO) << "send ping " << tag("id", id) << tag("cnt", cnt); + } + string text = PSTRING() << "PING: " << cnt; + send_message(id, std::move(text)); + } + void send_message(int id, string text) { + auto random_id = Random::secure_int64(); + LOG(INFO) << "send message: " << tag("id", id) << tag("text", text) << tag("random_id", random_id); + sent_messages_[random_id] = Message{id, text}; + send_closure(get_by_id(id), &SecretChatProxy::send_message, + secret_api::make_object<secret_api::decryptedMessage>(0, random_id, 0, text, Auto(), Auto(), Auto(), + Auto(), 0)); + } + void process_net_query(my_api::messages_sendEncryptedService &&message, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + process_net_query_send_enrypted(std::move(message.data_), std::move(net_query), std::move(callback)); + } + void process_net_query(my_api::messages_sendEncrypted &&message, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + process_net_query_send_enrypted(std::move(message.data_), std::move(net_query), std::move(callback)); + } + void process_net_query_send_enrypted(BufferSlice data, NetQueryPtr net_query, + ActorShared<NetQueryCallback> callback) { + my_api::messages_sentEncryptedMessage sent_message; + sent_message.date_ = 0; + auto storer = TLObjectStorer<my_api::messages_sentEncryptedMessage>(sent_message); + BufferSlice answer(storer.size()); + storer.store(answer.as_slice().ubegin()); + net_query->set_ok(std::move(answer)); + send_closure(std::move(callback), &NetQueryCallback::on_result, std::move(net_query)); + + // We can't loose updates yet :( + auto crc = crc64(data.as_slice()); + LOG(INFO) << "send SecretChatProxy::add_inbound_message" << tag("crc", crc); + send_closure(to(), &SecretChatProxy::add_inbound_message, narrow_cast<int32>(3 - get_link_token()), std::move(data), + crc); + } + + int32 last_ping_ = std::numeric_limits<int32>::max(); + void on_inbound_message(string message, Promise<> promise) { + promise.set_value(Unit()); + LOG(INFO) << "GOT INBOUND MESSAGE: " << message << " " << get_link_token(); + int32 cnt; + int x = std::sscanf(message.c_str(), "PING: %d", &cnt); + if (x != 1) { + return; + } + if (cnt == 0) { + Scheduler::instance()->finish(); + *status_ = Status::OK(); + return; + } + if (cnt >= last_ping_) { + return; + } + last_ping_ = cnt; + send_ping(narrow_cast<int32>(get_link_token()), cnt - 1); + } + void on_send_message_error(int64 random_id, Status error, Promise<> promise) { + promise.set_value(Unit()); + LOG(INFO) << "on_send_message_error: " << tag("random_id", random_id) << error; + auto it = sent_messages_.find(random_id); + if (it == sent_messages_.end()) { + LOG(INFO) << "TODO: try fix errors about message after it is sent"; + return; + } + CHECK(it != sent_messages_.end()); + auto message = it->second; + // sent_messages_.erase(it); + send_message(message.id, message.text); + } + void on_send_message_ok(int64 random_id, Promise<> promise) { + promise.set_value(Unit()); + LOG(INFO) << "on_send_message_ok: " << tag("random_id", random_id); + auto it = sent_messages_.find(random_id); + if (it == sent_messages_.end()) { + LOG(INFO) << "TODO: try fix errors about message after it is sent"; + return; + } + CHECK(it != sent_messages_.end()); + // sent_messages_.erase(it); + } + + private: + Status *status_; + ActorOwn<SecretChatProxy> alice_; + ActorOwn<SecretChatProxy> bob_; + struct Message { + int32 id; + string text; + }; + std::map<int64, Message> sent_messages_; + + void hangup_shared() override { + LOG(INFO) << "GOT HANGUP: " << get_link_token(); + send_closure(from(), &SecretChatProxy::on_closed); + } +}; + +void FakeSecretChatContext::send_net_query(NetQueryPtr query, ActorShared<NetQueryCallback> callback, bool ordered) { + send_closure(master_, &Master::send_net_query, std::move(query), std::move(callback), ordered); +} +void FakeSecretChatContext::on_inbound_message(UserId user_id, MessageId message_id, int32 date, + tl_object_ptr<telegram_api::encryptedFile> file, + tl_object_ptr<secret_api::decryptedMessage> message, Promise<> promise) { + send_closure(master_, &Master::on_inbound_message, message->message_, std::move(promise)); +} +void FakeSecretChatContext::on_send_message_error(int64 random_id, Status error, Promise<> promise) { + send_closure(master_, &Master::on_send_message_error, random_id, std::move(error), std::move(promise)); +} +void FakeSecretChatContext::on_send_message_ack(int64 random_id) { +} +void FakeSecretChatContext::on_send_message_ok(int64 random_id, MessageId message_id, int32 date, + tl_object_ptr<telegram_api::EncryptedFile> file, Promise<> promise) { + send_closure(master_, &Master::on_send_message_ok, random_id, std::move(promise)); +} +void FakeSecretChatContext::on_delete_messages(std::vector<int64> random_id, Promise<> promise) { + promise.set_value(Unit()); +} +void FakeSecretChatContext::on_flush_history(MessageId, Promise<> promise) { + promise.set_error(Status::Error("unsupported")); +} +void FakeSecretChatContext::on_read_message(int64, Promise<> promise) { + promise.set_error(Status::Error("unsupported")); +} + +TEST(Secret, go) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(ERROR)); + ConcurrentScheduler sched; + int threads_n = 0; + sched.init(threads_n); + + Status result; + sched.create_actor_unsafe<Master>(0, "HandshakeTestActor", &result).release(); + sched.start(); + while (sched.run_main(10)) { + // empty; + } + sched.finish(); + + if (result.is_error()) { + LOG(ERROR) << result; + } + ASSERT_TRUE(result.is_ok()); +} + +} // namespace td diff --git a/libs/tdlib/td/test/string_cleaning.cpp b/libs/tdlib/td/test/string_cleaning.cpp new file mode 100644 index 0000000000..6fbd8150a4 --- /dev/null +++ b/libs/tdlib/td/test/string_cleaning.cpp @@ -0,0 +1,107 @@ +// +// 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/telegram/misc.h" + +#include "td/utils/Slice.h" +#include "td/utils/tests.h" + +REGISTER_TESTS(string_cleaning); + +using namespace td; + +TEST(StringCleaning, clean_name) { + ASSERT_EQ("@mention", clean_name("@mention", 1000000)); + ASSERT_EQ("@mention", clean_name(" @mention ", 1000000)); + ASSERT_EQ("@MENTION", clean_name("@MENTION", 1000000)); + ASSERT_EQ("ЛШТШФУМ", clean_name("ЛШТШФУМ", 1000000)); + ASSERT_EQ("....", clean_name("....", 1000000)); + ASSERT_EQ(". ASD ..", clean_name(". ASD ..", 1000000)); + ASSERT_EQ(". ASD", clean_name(". ASD ..", 10)); + ASSERT_EQ(". ASD", clean_name(".\n\n\nASD\n\n\n..", 10)); + ASSERT_EQ("", clean_name("\n\n\n\n\n\n", 1000000)); + ASSERT_EQ("", clean_name("\xC2\xA0\xC2\xA0\xC2\xA0\xC2\xA0\xC2\xA0\n\n\n\n\n\n \n\xC2\xA0 \xC2\xA0 \n", 100000)); + ASSERT_EQ("abc", clean_name("\xC2\xA0\xC2\xA0" + "abc\xC2\xA0\xC2\xA0\xC2\xA0\xC2\xA0", + 1000000)); +}; + +TEST(StringCleaning, clean_username) { + ASSERT_EQ("@mention", clean_username("@mention")); + ASSERT_EQ("@mention", clean_username(" @mention ")); + ASSERT_EQ("@mention", clean_username("@MENTION")); + ASSERT_EQ("ЛШТШФУМ", clean_username("ЛШТШФУМ")); + ASSERT_EQ("", clean_username("....")); + ASSERT_EQ("asd", clean_username(". ASD ..")); +}; + +static void check_clean_input_string(string str, string expected, bool expected_result) { + auto result = clean_input_string(str); + ASSERT_EQ(expected_result, result); + if (result) { + ASSERT_EQ(expected, str); + } +} + +TEST(StringCleaning, clean_input_string) { + check_clean_input_string("/abc", "/abc", true); + check_clean_input_string(string(50000, 'a'), string(34996, 'a'), true); + check_clean_input_string("\xff", "", false); + check_clean_input_string("\xc0\x80", "", false); + check_clean_input_string("\xd0", "", false); + check_clean_input_string("\xe0\xaf", "", false); + check_clean_input_string("\xf0\xa6", "", false); + check_clean_input_string("\xf0\xa6\x88", "", false); + check_clean_input_string("\xf4\x8f\xbf\xbf", "\xf4\x8f\xbf\xbf", true); + check_clean_input_string("\xf4\x8f\xbf\xc0", "", false); + check_clean_input_string("\r\r\r\r\r\r\r", "", true); + check_clean_input_string("\r\n\r\n\r\n\r\n\r\n\r\n\r", "\n\n\n\n\n\n", true); + check_clean_input_string(Slice("\x00\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f\x10\x11\x12\x13\x14" + "\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f\x20\x21") + .str(), + " \x0a \x21", true); + check_clean_input_string( + "\xe2\x80\xa7\xe2\x80\xa8\xe2\x80\xa9\xe2\x80\xaa\xe2\x80\xab\xe2\x80\xac\xe2\x80\xad\xe2\x80\xae\xe2\x80\xaf", + "\xe2\x80\xa7\xe2\x80\xaf", true); + check_clean_input_string("\xcc\xb3\xcc\xbf\xcc\x8a", "", true); +} + +static void check_strip_empty_characters(string str, size_t max_length, string expected) { + ASSERT_EQ(expected, strip_empty_characters(str, max_length)); +} + +TEST(StringCleaning, strip_empty_characters) { + check_strip_empty_characters("/abc", 4, "/abc"); + check_strip_empty_characters("/abc", 3, "/ab"); + check_strip_empty_characters("/abc", 0, ""); + check_strip_empty_characters("/abc", 10000000, "/abc"); + string spaces = + u8"\u1680\u180E\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200A\u200B\u202F\u205F\u3000\uFEFF" + u8"\uFFFC\uFFFC"; + string spaces_replace = " "; + string empty = "\xE2\x80\x8C\xE2\x80\x8D\xE2\x80\xAE\xC2\xA0\xC2\xA0"; + + check_strip_empty_characters(spaces, 1000000, ""); + check_strip_empty_characters(empty, 1000000, ""); + check_strip_empty_characters(empty + "a", 1000000, empty + "a"); + check_strip_empty_characters(spaces + empty + spaces + "abc" + spaces, 1000000, empty + spaces_replace + "abc"); + check_strip_empty_characters(spaces + spaces + empty + spaces + spaces + empty + empty, 1000000, + empty + spaces_replace + spaces_replace + empty + empty); + check_strip_empty_characters("\r\r\r\r\r\r\r", 1000000, ""); + check_strip_empty_characters("\r\n\r\n\r\n\r\n\r\n\r\n\r", 1000000, ""); + check_strip_empty_characters(Slice(" \t\r\n\0\va\v\0\n\r\t ").str(), 1000000, "a"); + check_strip_empty_characters( + Slice("\x00\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f\x10\x11\x12\x13\x14" + "\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f\x20\x21") + .str(), + 1000000, + "\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0a\x0b\x0c\x0d\x0e\x0f\x10\x11\x12\x13\x14" + "\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f\x20\x21"); + check_strip_empty_characters("\xcc\xb3\xcc\xbf\xcc\x8a", 2, "\xcc\xb3\xcc\xbf"); + check_strip_empty_characters( + "\xe2\x80\xa7\xe2\x80\xa8\xe2\x80\xa9\xe2\x80\xaa\xe2\x80\xab\xe2\x80\xac\xe2\x80\xad\xe2\x80\xae", 3, + "\xe2\x80\xa7\xe2\x80\xa8\xe2\x80\xa9"); +} diff --git a/libs/tdlib/td/test/tdclient.cpp b/libs/tdlib/td/test/tdclient.cpp new file mode 100644 index 0000000000..7cef0fcbc0 --- /dev/null +++ b/libs/tdlib/td/test/tdclient.cpp @@ -0,0 +1,837 @@ +// +// 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 "data.h" + +#include "td/actor/PromiseFuture.h" + +#include "td/telegram/ClientActor.h" + +#include "td/telegram/td_api.h" + +#include "td/utils/base64.h" +#include "td/utils/BufferedFd.h" +#include "td/utils/filesystem.h" +#include "td/utils/format.h" +#include "td/utils/logging.h" +#include "td/utils/misc.h" +#include "td/utils/port/FileFd.h" +#include "td/utils/port/path.h" +#include "td/utils/Random.h" +#include "td/utils/Slice.h" +#include "td/utils/Status.h" +#include "td/utils/tests.h" + +#include <cstdio> +#include <functional> +#include <map> +#include <memory> +#include <utility> + +REGISTER_TESTS(tdclient); + +namespace td { + +template <class T> +static void check_td_error(T &result) { + CHECK(result->get_id() != td_api::error::ID) << to_string(result); +} + +static void rmrf(CSlice path) { + td::walk_path(path, [](CSlice path, bool is_dir) { + if (is_dir) { + td::rmdir(path).ignore(); + } else { + td::unlink(path).ignore(); + } + }); +} + +class TestClient : public Actor { + public: + explicit TestClient(string name) : name_(std::move(name)) { + } + struct Update { + uint64 id; + tl_object_ptr<td_api::Object> object; + Update(uint64 id, tl_object_ptr<td_api::Object> object) : id(id), object(std::move(object)) { + } + }; + class Listener { + public: + Listener() = default; + Listener(const Listener &) = delete; + Listener &operator=(const Listener &) = delete; + Listener(Listener &&) = delete; + Listener &operator=(Listener &&) = delete; + virtual ~Listener() = default; + virtual void start_listen(TestClient *client) { + } + virtual void stop_listen() { + } + virtual void on_update(std::shared_ptr<Update> update) = 0; + }; + void close(Promise<> close_promise) { + close_promise_ = std::move(close_promise); + td_.reset(); + } + + unique_ptr<TdCallback> make_td_callback() { + class TdCallbackImpl : public TdCallback { + public: + explicit TdCallbackImpl(ActorId<TestClient> client) : client_(client) { + } + void on_result(uint64 id, tl_object_ptr<td_api::Object> result) override { + send_closure(client_, &TestClient::on_result, id, std::move(result)); + } + void on_error(uint64 id, tl_object_ptr<td_api::error> error) override { + send_closure(client_, &TestClient::on_error, id, std::move(error)); + } + void on_closed() override { + send_closure(client_, &TestClient::on_closed); + } + + private: + ActorId<TestClient> client_; + }; + return make_unique<TdCallbackImpl>(actor_id(this)); + } + + void add_listener(std::unique_ptr<Listener> listener) { + auto *ptr = listener.get(); + listeners_.push_back(std::move(listener)); + ptr->start_listen(this); + } + void remove_listener(Listener *listener) { + pending_remove_.push_back(listener); + } + void do_pending_remove_listeners() { + for (auto listener : pending_remove_) { + do_remove_listener(listener); + } + pending_remove_.clear(); + } + void do_remove_listener(Listener *listener) { + for (size_t i = 0; i < listeners_.size(); i++) { + if (listeners_[i].get() == listener) { + listener->stop_listen(); + listeners_.erase(listeners_.begin() + i); + break; + } + } + } + + void on_result(uint64 id, tl_object_ptr<td_api::Object> result) { + on_update(std::make_shared<Update>(id, std::move(result))); + } + void on_error(uint64 id, tl_object_ptr<td_api::error> error) { + on_update(std::make_shared<Update>(id, std::move(error))); + } + void on_update(std::shared_ptr<Update> update) { + for (auto &listener : listeners_) { + listener->on_update(update); + } + do_pending_remove_listeners(); + } + + void on_closed() { + stop(); + } + + void start_up() override { + rmrf(name_); + set_context(std::make_shared<td::ActorContext>()); + set_tag(name_); + LOG(INFO) << "START UP!"; + + td_ = create_actor<ClientActor>("Td-proxy", make_td_callback()); + } + + ActorOwn<ClientActor> td_; + + string name_; + + private: + std::vector<std::unique_ptr<Listener>> listeners_; + std::vector<Listener *> pending_remove_; + + Promise<> close_promise_; +}; + +class Task : public TestClient::Listener { + public: + void on_update(std::shared_ptr<TestClient::Update> update) override { + auto it = sent_queries_.find(update->id); + if (it != sent_queries_.end()) { + it->second(std::move(update->object)); + sent_queries_.erase(it); + } + process_update(update); + } + void start_listen(TestClient *client) override { + client_ = client; + start_up(); + } + virtual void process_update(std::shared_ptr<TestClient::Update> update) { + } + + template <class F> + void send_query(tl_object_ptr<td_api::Function> function, F callback) { + auto id = current_query_id_++; + sent_queries_[id] = std::forward<F>(callback); + send_closure(client_->td_, &ClientActor::request, id, std::move(function)); + } + + protected: + std::map<uint64, std::function<void(tl_object_ptr<td_api::Object>)>> sent_queries_; + TestClient *client_; + uint64 current_query_id_ = 1; + + virtual void start_up() { + } + void stop() { + client_->remove_listener(this); + } +}; + +class DoAuthentication : public Task { + public: + DoAuthentication(string name, string phone, string code, Promise<> promise) + : name_(std::move(name)), phone_(std::move(phone)), code_(std::move(code)), promise_(std::move(promise)) { + } + void start_up() override { + send_query(make_tl_object<td_api::getAuthorizationState>(), + [this](auto res) { this->process_authorization_state(std::move(res)); }); + } + void process_authorization_state(tl_object_ptr<td_api::Object> authorization_state) { + start_flag_ = true; + tl_object_ptr<td_api::Function> function; + switch (authorization_state->get_id()) { + case td_api::authorizationStateWaitEncryptionKey::ID: + function = make_tl_object<td_api::checkDatabaseEncryptionKey>(); + break; + case td_api::authorizationStateWaitPhoneNumber::ID: + function = make_tl_object<td_api::setAuthenticationPhoneNumber>(phone_, false, true); + break; + case td_api::authorizationStateWaitCode::ID: + function = make_tl_object<td_api::checkAuthenticationCode>(code_, name_, ""); + break; + case td_api::authorizationStateWaitTdlibParameters::ID: { + auto parameters = td_api::make_object<td_api::tdlibParameters>(); + parameters->use_test_dc_ = true; + parameters->database_directory_ = name_ + TD_DIR_SLASH; + parameters->use_message_database_ = true; + parameters->use_secret_chats_ = true; + parameters->api_id_ = 94575; + parameters->api_hash_ = "a3406de8d171bb422bb6ddf3bbd800e2"; + parameters->system_language_code_ = "en"; + parameters->device_model_ = "Desktop"; + parameters->system_version_ = "Unknown"; + parameters->application_version_ = "tdclient-test"; + parameters->ignore_file_names_ = false; + parameters->enable_storage_optimizer_ = true; + function = td_api::make_object<td_api::setTdlibParameters>(std::move(parameters)); + break; + } + case td_api::authorizationStateReady::ID: + on_authorization_ready(); + return; + default: + CHECK(false) << "Unexpected authorization state " << to_string(authorization_state); + } + send_query(std::move(function), [this](auto res) { CHECK(res->get_id() == td_api::ok::ID) << to_string(res); }); + } + void on_authorization_ready() { + LOG(INFO) << "GOT AUTHORIZED"; + stop(); + } + + private: + string name_; + string phone_; + string code_; + Promise<> promise_; + bool start_flag_{false}; + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (!start_flag_) { + return; + } + if (!update->object) { + return; + } + if (update->object->get_id() == td_api::updateAuthorizationState::ID) { + auto o = std::move(update->object); + process_authorization_state(std::move(static_cast<td_api::updateAuthorizationState &>(*o).authorization_state_)); + } + } +}; + +class SetUsername : public Task { + public: + SetUsername(string username, Promise<> promise) : username_(std::move(username)), promise_(std::move(promise)) { + } + + private: + string username_; + Promise<> promise_; + int32 self_id_; + string tag_; + + void start_up() override { + send_query(make_tl_object<td_api::getMe>(), [this](auto res) { this->process_me_user(std::move(res)); }); + } + + void process_me_user(tl_object_ptr<td_api::Object> res) { + CHECK(res->get_id() == td_api::user::ID); + auto user = move_tl_object_as<td_api::user>(res); + self_id_ = user->id_; + if (user->username_ != username_) { + LOG(INFO) << "SET USERNAME: " << username_; + send_query(make_tl_object<td_api::setUsername>(username_), [this](auto res) { + CHECK(res->get_id() == td_api::ok::ID); + this->send_self_message(); + }); + } else { + send_self_message(); + } + } + void send_self_message() { + tag_ = PSTRING() << format::as_hex(Random::secure_int64()); + + send_query(make_tl_object<td_api::createPrivateChat>(self_id_, false), [this](auto res) { + CHECK(res->get_id() == td_api::chat::ID); + auto chat = move_tl_object_as<td_api::chat>(res); + this->send_query( + make_tl_object<td_api::sendMessage>( + chat->id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageText>( + make_tl_object<td_api::formattedText>(PSTRING() << tag_ << " INIT", Auto()), false, false)), + [](auto res) {}); + }); + } + + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (!update->object) { + return; + } + if (update->object->get_id() == td_api::updateMessageSendSucceeded::ID) { + auto updateNewMessage = move_tl_object_as<td_api::updateMessageSendSucceeded>(update->object); + auto &message = updateNewMessage->message_; + if (message->content_->get_id() == td_api::messageText::ID) { + auto messageText = move_tl_object_as<td_api::messageText>(message->content_); + auto text = messageText->text_->text_; + if (text.substr(0, tag_.size()) == tag_) { + LOG(INFO) << "GOT SELF MESSAGE"; + return stop(); + } + } + } + } +}; + +class CheckTestA : public Task { + public: + CheckTestA(string tag, Promise<> promise) : tag_(std::move(tag)), promise_(std::move(promise)) { + } + + private: + string tag_; + Promise<> promise_; + string previous_text_; + int cnt_ = 20; + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (update->object->get_id() == td_api::updateNewMessage::ID) { + auto updateNewMessage = move_tl_object_as<td_api::updateNewMessage>(update->object); + auto &message = updateNewMessage->message_; + if (message->content_->get_id() == td_api::messageText::ID) { + auto messageText = move_tl_object_as<td_api::messageText>(message->content_); + auto text = messageText->text_->text_; + if (text.substr(0, tag_.size()) == tag_) { + CHECK(text > previous_text_) << tag("now", text) << tag("previous", previous_text_); + previous_text_ = text; + cnt_--; + LOG(INFO) << "GOT " << tag("text", text) << tag("left", cnt_); + if (cnt_ == 0) { + return stop(); + } + } + } + } + } +}; + +class TestA : public Task { + public: + TestA(string tag, string username) : tag_(std::move(tag)), username_(std::move(username)) { + } + void start_up() override { + send_query(make_tl_object<td_api::searchPublicChat>(username_), [this](auto res) { + CHECK(res->get_id() == td_api::chat::ID); + auto chat = move_tl_object_as<td_api::chat>(res); + for (int i = 0; i < 20; i++) { + this->send_query(make_tl_object<td_api::sendMessage>( + chat->id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageText>( + make_tl_object<td_api::formattedText>(PSTRING() << tag_ << " " << (1000 + i), Auto()), + false, false)), + [&](auto res) { this->stop(); }); + } + }); + } + + private: + string tag_; + string username_; +}; + +class TestSecretChat : public Task { + public: + TestSecretChat(string tag, string username) : tag_(std::move(tag)), username_(std::move(username)) { + } + + void start_up() override { + auto f = [this](auto res) { + CHECK(res->get_id() == td_api::chat::ID); + auto chat = move_tl_object_as<td_api::chat>(res); + this->chat_id_ = chat->id_; + this->secret_chat_id_ = move_tl_object_as<td_api::chatTypeSecret>(chat->type_)->secret_chat_id_; + }; + send_query(make_tl_object<td_api::searchPublicChat>(username_), [this, f = std::move(f)](auto res) mutable { + CHECK(res->get_id() == td_api::chat::ID); + auto chat = move_tl_object_as<td_api::chat>(res); + CHECK(chat->type_->get_id() == td_api::chatTypePrivate::ID); + auto info = move_tl_object_as<td_api::chatTypePrivate>(chat->type_); + this->send_query(make_tl_object<td_api::createNewSecretChat>(info->user_id_), std::move(f)); + }); + } + + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (!update->object) { + return; + } + if (update->object->get_id() == td_api::updateSecretChat::ID) { + auto update_secret_chat = move_tl_object_as<td_api::updateSecretChat>(update->object); + if (update_secret_chat->secret_chat_->id_ != secret_chat_id_ || + update_secret_chat->secret_chat_->state_->get_id() != td_api::secretChatStateReady::ID) { + return; + } + LOG(INFO) << "SEND ENCRYPTED MESSAGES"; + for (int i = 0; i < 20; i++) { + send_query(make_tl_object<td_api::sendMessage>( + chat_id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageText>( + make_tl_object<td_api::formattedText>(PSTRING() << tag_ << " " << (1000 + i), Auto()), false, + false)), + [](auto res) {}); + } + } + } + + private: + string tag_; + string username_; + int64 secret_chat_id_ = 0; + int64 chat_id_ = 0; +}; + +class TestFileGenerated : public Task { + public: + TestFileGenerated(string tag, string username) : tag_(std::move(tag)), username_(std::move(username)) { + } + + void start_up() override { + } + + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (!update->object) { + return; + } + if (update->object->get_id() == td_api::updateNewMessage::ID) { + auto updateNewMessage = move_tl_object_as<td_api::updateNewMessage>(update->object); + auto &message = updateNewMessage->message_; + chat_id_ = message->chat_id_; + if (message->content_->get_id() == td_api::messageText::ID) { + auto messageText = move_tl_object_as<td_api::messageText>(message->content_); + auto text = messageText->text_->text_; + if (text.substr(0, tag_.size()) == tag_) { + if (text.substr(tag_.size() + 1) == "ONE_FILE") { + return one_file(); + } + } + } + } else if (update->object->get_id() == td_api::updateFileGenerationStart::ID) { + auto info = move_tl_object_as<td_api::updateFileGenerationStart>(update->object); + generate_file(info->generation_id_, info->original_path_, info->destination_path_, info->conversion_); + } else if (update->object->get_id() == td_api::updateFile::ID) { + auto file = move_tl_object_as<td_api::updateFile>(update->object); + LOG(INFO) << to_string(file); + } + } + void one_file() { + LOG(ERROR) << "Start ONE_FILE test"; + string file_path = string("test_documents") + TD_DIR_SLASH + "a.txt"; + mkpath(file_path).ensure(); + auto raw_file = + FileFd::open(file_path, FileFd::Flags::Create | FileFd::Flags::Truncate | FileFd::Flags::Write).move_as_ok(); + auto file = BufferedFd<FileFd>(std::move(raw_file)); + for (int i = 1; i < 100000; i++) { + file.write(PSLICE() << i << "\n").ensure(); + } + file.flush_write().ensure(); // important + file.close(); + this->send_query(make_tl_object<td_api::sendMessage>( + chat_id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageDocument>( + make_tl_object<td_api::inputFileGenerated>(file_path, "square", 0), + make_tl_object<td_api::inputThumbnail>( + make_tl_object<td_api::inputFileGenerated>(file_path, "thumbnail", 0), 0, 0), + make_tl_object<td_api::formattedText>(tag_, Auto()))), + [](auto res) { check_td_error(res); }); + + this->send_query( + make_tl_object<td_api::sendMessage>(chat_id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageDocument>( + make_tl_object<td_api::inputFileGenerated>(file_path, "square", 0), + nullptr, make_tl_object<td_api::formattedText>(tag_, Auto()))), + [](auto res) { check_td_error(res); }); + } + + friend class GenerateFile; + class GenerateFile : public Actor { + public: + GenerateFile(Task *parent, int64 id, string original_path, string destination_path, string conversion) + : parent_(parent) + , id_(id) + , original_path_(std::move(original_path)) + , destination_path_(std::move(destination_path)) + , conversion_(std::move(conversion)) { + } + + private: + Task *parent_; + int64 id_; + string original_path_; + string destination_path_; + string conversion_; + + FILE *from = nullptr; + FILE *to = nullptr; + + void start_up() override { + from = std::fopen(original_path_.c_str(), "rb"); + CHECK(from); + to = std::fopen(destination_path_.c_str(), "wb"); + CHECK(to); + yield(); + } + void loop() override { + int cnt = 0; + while (true) { + uint32 x; + auto r = std::fscanf(from, "%u", &x); + if (r != 1) { + return stop(); + } + std::fprintf(to, "%u\n", x * x); + if (++cnt >= 10000) { + break; + } + } + auto ready = std::ftell(to); + LOG(ERROR) << "READY: " << ready; + parent_->send_query(make_tl_object<td_api::setFileGenerationProgress>( + id_, 1039823 /*yeah, exact size of this file*/, narrow_cast<int32>(ready)), + [](auto result) { check_td_error(result); }); + set_timeout_in(0.02); + } + void tear_down() override { + std::fclose(from); + std::fclose(to); + parent_->send_query(make_tl_object<td_api::finishFileGeneration>(id_, nullptr), + [](auto result) { check_td_error(result); }); + } + }; + void generate_file(int64 id, string original_path, string destination_path, string conversion) { + LOG(ERROR) << "Generate file " << tag("id", id) << tag("original_path", original_path) + << tag("destination_path", destination_path) << tag("conversion", conversion); + if (conversion == "square") { + create_actor<GenerateFile>("GenerateFile", this, id, original_path, destination_path, conversion).release(); + } else if (conversion == "thumbnail") { + write_file(destination_path, base64url_decode(Slice(thumbnail, thumbnail_size)).ok()).ensure(); + this->send_query(make_tl_object<td_api::finishFileGeneration>(id, nullptr), + [](auto result) { check_td_error(result); }); + } else { + LOG(FATAL) << "unknown " << tag("conversion", conversion); + } + } + + private: + string tag_; + string username_; + int64 chat_id_ = 0; +}; + +class CheckTestC : public Task { + public: + CheckTestC(string username, string tag, Promise<> promise) + : username_(std::move(username)), tag_(std::move(tag)), promise_(std::move(promise)) { + } + + void start_up() override { + send_query(make_tl_object<td_api::searchPublicChat>(username_), [this](auto res) { + CHECK(res->get_id() == td_api::chat::ID); + auto chat = move_tl_object_as<td_api::chat>(res); + chat_id_ = chat->id_; + this->one_file(); + }); + } + + private: + string username_; + string tag_; + Promise<> promise_; + int64 chat_id_; + + void one_file() { + this->send_query( + make_tl_object<td_api::sendMessage>( + chat_id_, 0, false, false, nullptr, + make_tl_object<td_api::inputMessageText>( + make_tl_object<td_api::formattedText>(PSTRING() << tag_ << " ONE_FILE", Auto()), false, false)), + [](auto res) { check_td_error(res); }); + } + + void process_update(std::shared_ptr<TestClient::Update> update) override { + if (!update->object) { + return; + } + if (update->object->get_id() == td_api::updateNewMessage::ID) { + auto updateNewMessage = move_tl_object_as<td_api::updateNewMessage>(update->object); + auto &message = updateNewMessage->message_; + if (message->content_->get_id() == td_api::messageDocument::ID) { + auto messageDocument = move_tl_object_as<td_api::messageDocument>(message->content_); + auto text = messageDocument->caption_->text_; + if (text.substr(0, tag_.size()) == tag_) { + file_id_to_check_ = messageDocument->document_->document_->id_; + LOG(ERROR) << "GOT FILE " << to_string(messageDocument->document_->document_); + this->send_query(make_tl_object<td_api::downloadFile>(file_id_to_check_, 1), + [](auto res) { check_td_error(res); }); + } + } + } else if (update->object->get_id() == td_api::updateFile::ID) { + auto updateFile = move_tl_object_as<td_api::updateFile>(update->object); + if (updateFile->file_->id_ == file_id_to_check_ && (updateFile->file_->local_->is_downloading_completed_)) { + check_file(updateFile->file_->local_->path_); + } + } + } + + void check_file(CSlice path) { + FILE *from = std::fopen(path.c_str(), "rb"); + CHECK(from); + uint32 x; + uint32 y = 1; + while (std::fscanf(from, "%u", &x) == 1) { + CHECK(x == y * y); + y++; + } + std::fclose(from); + stop(); + } + int32 file_id_to_check_ = 0; +}; + +class LoginTestActor : public Actor { + public: + explicit LoginTestActor(Status *status) : status_(status) { + *status_ = Status::OK(); + } + + private: + Status *status_; + ActorOwn<TestClient> alice_; + ActorOwn<TestClient> bob_; + + string alice_phone_ = "9996636437"; + string bob_phone_ = "9996636438"; + string alice_username_ = "alice_" + alice_phone_; + string bob_username_ = "bob_" + bob_phone_; + + string stage_name_; + + void begin_stage(string stage_name, double timeout) { + LOG(WARNING) << "Begin stage '" << stage_name << "'"; + stage_name_ = std::move(stage_name); + set_timeout_in(timeout); + } + + void start_up() override { + begin_stage("Logging in", 160); + alice_ = create_actor<TestClient>("AliceClient", "alice"); + bob_ = create_actor<TestClient>("BobClient", "bob"); + + send_closure(alice_, &TestClient::add_listener, + std::make_unique<DoAuthentication>( + "alice", alice_phone_, "33333", + PromiseCreator::event(self_closure(this, &LoginTestActor::start_up_fence_dec)))); + + send_closure(bob_, &TestClient::add_listener, + std::make_unique<DoAuthentication>( + "bob", bob_phone_, "33333", + PromiseCreator::event(self_closure(this, &LoginTestActor::start_up_fence_dec)))); + } + + int start_up_fence_ = 3; + void start_up_fence_dec() { + --start_up_fence_; + if (start_up_fence_ == 0) { + init(); + } else if (start_up_fence_ == 1) { + return init(); + class WaitActor : public Actor { + public: + WaitActor(double timeout, Promise<> promise) : timeout_(timeout), promise_(std::move(promise)) { + } + void start_up() override { + set_timeout_in(timeout_); + } + void timeout_expired() override { + stop(); + } + + private: + double timeout_; + Promise<> promise_; + }; + create_actor<WaitActor>("WaitActor", 2, + PromiseCreator::event(self_closure(this, &LoginTestActor::start_up_fence_dec))) + .release(); + } + } + + void init() { + send_closure(alice_, &TestClient::add_listener, + std::make_unique<SetUsername>( + alice_username_, PromiseCreator::event(self_closure(this, &LoginTestActor::init_fence_dec)))); + send_closure(bob_, &TestClient::add_listener, + std::make_unique<SetUsername>( + bob_username_, PromiseCreator::event(self_closure(this, &LoginTestActor::init_fence_dec)))); + } + + int init_fence_ = 2; + void init_fence_dec() { + if (--init_fence_ == 0) { + test_a(); + } + } + + int32 test_a_fence_ = 2; + void test_a_fence() { + if (--test_a_fence_ == 0) { + test_b(); + } + } + + void test_a() { + begin_stage("Ready to create chats", 80); + string alice_tag = PSTRING() << format::as_hex(Random::secure_int64()); + string bob_tag = PSTRING() << format::as_hex(Random::secure_int64()); + + send_closure(bob_, &TestClient::add_listener, + std::make_unique<CheckTestA>( + alice_tag, PromiseCreator::event(self_closure(this, &LoginTestActor::test_a_fence)))); + send_closure(alice_, &TestClient::add_listener, + std::make_unique<CheckTestA>( + bob_tag, PromiseCreator::event(self_closure(this, &LoginTestActor::test_a_fence)))); + + send_closure(alice_, &TestClient::add_listener, std::make_unique<TestA>(alice_tag, bob_username_)); + send_closure(bob_, &TestClient::add_listener, std::make_unique<TestA>(bob_tag, alice_username_)); + // send_closure(alice_, &TestClient::add_listener, std::make_unique<TestChat>(bob_username_)); + } + + void timeout_expired() override { + LOG(FATAL) << "Timeout expired in stage '" << stage_name_ << "'"; + } + + int32 test_b_fence_ = 1; + void test_b_fence() { + if (--test_b_fence_ == 0) { + test_c(); + } + } + + int32 test_c_fence_ = 1; + void test_c_fence() { + if (--test_c_fence_ == 0) { + finish(); + } + } + + void test_b() { + begin_stage("Create secret chat", 40); + string tag = PSTRING() << format::as_hex(Random::secure_int64()); + + send_closure( + bob_, &TestClient::add_listener, + std::make_unique<CheckTestA>(tag, PromiseCreator::event(self_closure(this, &LoginTestActor::test_b_fence)))); + send_closure(alice_, &TestClient::add_listener, std::make_unique<TestSecretChat>(tag, bob_username_)); + } + + void test_c() { + begin_stage("Send generated file", 240); + string tag = PSTRING() << format::as_hex(Random::secure_int64()); + + send_closure(bob_, &TestClient::add_listener, + std::make_unique<CheckTestC>( + alice_username_, tag, PromiseCreator::event(self_closure(this, &LoginTestActor::test_c_fence)))); + send_closure(alice_, &TestClient::add_listener, std::make_unique<TestFileGenerated>(tag, bob_username_)); + } + + int32 finish_fence_ = 2; + void finish_fence() { + finish_fence_--; + if (finish_fence_ == 0) { + Scheduler::instance()->finish(); + stop(); + } + } + void finish() { + send_closure(alice_, &TestClient::close, PromiseCreator::event(self_closure(this, &LoginTestActor::finish_fence))); + send_closure(bob_, &TestClient::close, PromiseCreator::event(self_closure(this, &LoginTestActor::finish_fence))); + } +}; + +class Tdclient_login : public td::Test { + public: + using Test::Test; + bool step() final { + if (!is_inited_) { + SET_VERBOSITY_LEVEL(VERBOSITY_NAME(DEBUG) + 2); + sched_.init(4); + sched_.create_actor_unsafe<LoginTestActor>(0, "LoginTestActor", &result_).release(); + sched_.start(); + is_inited_ = true; + } + + bool ret = sched_.run_main(10); + if (ret) { + return true; + } + sched_.finish(); + if (result_.is_error()) { + LOG(ERROR) << result_; + } + ASSERT_TRUE(result_.is_ok()); + return false; + } + + private: + bool is_inited_ = false; + ConcurrentScheduler sched_; + Status result_; +}; +Tdclient_login Tdclient_login("Tdclient_login"); +}; // namespace td diff --git a/libs/tdlib/td/test/tests_runner.cpp b/libs/tdlib/td/test/tests_runner.cpp new file mode 100644 index 0000000000..0edb186f0a --- /dev/null +++ b/libs/tdlib/td/test/tests_runner.cpp @@ -0,0 +1,18 @@ +// +// 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 "test/tests_runner.h" + +#include "test/TestsRunner.h" + +extern "C" { +void tests_runner_init(const char *dir) { + td::TestsRunner::init(dir); +} +void run_all_tests() { + td::TestsRunner::run_all_tests(); +} +} diff --git a/libs/tdlib/td/test/tests_runner.h b/libs/tdlib/td/test/tests_runner.h new file mode 100644 index 0000000000..3de566858b --- /dev/null +++ b/libs/tdlib/td/test/tests_runner.h @@ -0,0 +1,18 @@ +// +// 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) +// +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +void tests_runner_init(const char *dir); +void run_all_tests(); + +#ifdef __cplusplus +} +#endif |