diff --git a/.gitignore b/.gitignore index 6f6bb2408..4928acf93 100644 --- a/.gitignore +++ b/.gitignore @@ -18,6 +18,10 @@ /Telegram/Resources/art/sprite_125x.png /Telegram/Resources/art/sprite_150x.png /Telegram/Debug/ +/Telegram/Release/ +/Telegram/tests/ +/Telegram/gyp/tests/*.test +/Telegram/out/ /Telegram/*.user *.vcxproj* *.sln diff --git a/.gitmodules b/.gitmodules index 874ea8efc..288741058 100644 --- a/.gitmodules +++ b/.gitmodules @@ -7,3 +7,6 @@ [submodule "Telegram/ThirdParty/GSL"] path = Telegram/ThirdParty/GSL url = https://github.com/Microsoft/GSL.git +[submodule "Telegram/ThirdParty/Catch"] + path = Telegram/ThirdParty/Catch + url = https://github.com/philsquared/Catch diff --git a/README.md b/README.md index 4f8d88c27..7d8685fb4 100644 --- a/README.md +++ b/README.md @@ -37,6 +37,7 @@ The source code is published under GPLv3 with OpenSSL exception, the license is * Mapbox Variant ([BSD License](https://github.com/mapbox/variant/blob/master/LICENSE)) * Open Sans font ([Apache License 2.0](http://www.apache.org/licenses/LICENSE-2.0.html)) * Emoji alpha codes ([MIT License](https://github.com/emojione/emojione/blob/master/extras/alpha-codes/LICENSE.md)) +* Catch test framework ([Boost License](https://github.com/philsquared/Catch/blob/master/LICENSE.txt)) ## Build instructions diff --git a/Telegram/Patches/gyp.diff b/Telegram/Patches/gyp.diff index 13ddd9b7d..134ae28c0 100644 --- a/Telegram/Patches/gyp.diff +++ b/Telegram/Patches/gyp.diff @@ -1,33 +1,5 @@ -diff --git a/pylib/gyp/generator/cmake.py b/pylib/gyp/generator/cmake.py -index a2b9629..68d7020 100644 ---- a/pylib/gyp/generator/cmake.py -+++ b/pylib/gyp/generator/cmake.py -@@ -1070,6 +1070,23 @@ def WriteTarget(namer, qualified_target, target_dicts, build_dir, config_to_use, - - output.write(')\n') - -+ # Precompile header -+ precompiled_header = config.get('cmake_precompiled_header', '') -+ if precompiled_header: -+ precompiled_header_script = config.get('cmake_precompiled_header_script', '') -+ if not precompiled_header_script: -+ print ('ERROR: cmake_precompiled_header requires cmake_precompiled_header_script') -+ cmake_precompiled_header = NormjoinPath(path_from_cmakelists_to_gyp, precompiled_header) -+ cmake_precompiled_header_script = NormjoinPathForceCMakeSource(path_from_cmakelists_to_gyp, precompiled_header_script) -+ output.write('include(') -+ output.write(cmake_precompiled_header_script) -+ output.write(')\n') -+ output.write('add_precompiled_header(') -+ output.write(cmake_target_name) -+ output.write(' ') -+ output.write(cmake_precompiled_header) -+ output.write(')\n') -+ - UnsetVariable(output, 'TOOLSET') - UnsetVariable(output, 'TARGET') - diff --git a/pylib/gyp/generator/xcode.py b/pylib/gyp/generator/xcode.py -index db99d6a..f8398cc 100644 +index 0e3fb93..4c824ec 100644 --- a/pylib/gyp/generator/xcode.py +++ b/pylib/gyp/generator/xcode.py @@ -72,6 +72,10 @@ generator_additional_non_configuration_keys = [ @@ -41,7 +13,7 @@ index db99d6a..f8398cc 100644 'mac_bundle', 'mac_bundle_resources', 'mac_framework_headers', -@@ -772,6 +776,26 @@ def GenerateOutput(target_list, target_dicts, data, params): +@@ -761,6 +765,26 @@ def GenerateOutput(target_list, target_dicts, data, params): xcode_targets[qualified_target] = xct xcode_target_to_target_dict[xct] = spec @@ -68,3 +40,13 @@ index db99d6a..f8398cc 100644 spec_actions = spec.get('actions', []) spec_rules = spec.get('rules', []) +@@ -1130,7 +1154,8 @@ exit 1 + groups = [x for x in groups if not x.endswith('_excluded')] + for group in groups: + for item in rule.get(group, []): +- pbxp.AddOrGetFileInRootGroup(item) ++ concrete_item = ExpandXcodeVariables(item, rule_input_dict) ++ pbxp.AddOrGetFileInRootGroup(concrete_item) + + # Add "sources". + for source in spec.get('sources', []): diff --git a/Telegram/SourceFiles/base/flat_map.h b/Telegram/SourceFiles/base/flat_map.h index 6d4327726..d1dc28fd3 100644 --- a/Telegram/SourceFiles/base/flat_map.h +++ b/Telegram/SourceFiles/base/flat_map.h @@ -20,6 +20,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once +#include <deque> #include "base/optional.h" namespace base { diff --git a/Telegram/SourceFiles/base/flat_map_tests.cpp b/Telegram/SourceFiles/base/flat_map_tests.cpp new file mode 100644 index 000000000..88a83a2ee --- /dev/null +++ b/Telegram/SourceFiles/base/flat_map_tests.cpp @@ -0,0 +1,51 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#include "catch.hpp" + +#include "base/flat_map.h" +#include <string> + +using namespace std; + +TEST_CASE("flat_maps should keep items sorted by key", "[flat_map]") { + base::flat_map<int, string> v; + v.emplace(0, "a"); + v.emplace(5, "b"); + v.emplace(4, "d"); + v.emplace(2, "e"); + + auto checkSorted = [&] { + auto prev = v.begin(); + REQUIRE(prev != v.end()); + for (auto i = prev + 1; i != v.end(); prev = i, ++i) { + REQUIRE(prev->first < i->first); + } + }; + REQUIRE(v.size() == 4); + checkSorted(); + + SECTION("adding item puts it in the right position") { + v.emplace(3, "c"); + REQUIRE(v.size() == 5); + REQUIRE(v.find(3) != v.end()); + checkSorted(); + } +} diff --git a/Telegram/SourceFiles/base/flat_set.h b/Telegram/SourceFiles/base/flat_set.h index be75586b9..7e578d334 100644 --- a/Telegram/SourceFiles/base/flat_set.h +++ b/Telegram/SourceFiles/base/flat_set.h @@ -20,6 +20,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once +#include <deque> + namespace base { template <typename Type> diff --git a/Telegram/SourceFiles/base/flat_set_tests.cpp b/Telegram/SourceFiles/base/flat_set_tests.cpp new file mode 100644 index 000000000..675b0bdaf --- /dev/null +++ b/Telegram/SourceFiles/base/flat_set_tests.cpp @@ -0,0 +1,48 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#include "catch.hpp" + +#include "base/flat_set.h" + +TEST_CASE("flat_sets should keep items sorted", "[flat_set]") { + base::flat_set<int> v; + v.insert(0); + v.insert(5); + v.insert(4); + v.insert(2); + + auto checkSorted = [&] { + auto prev = v.begin(); + REQUIRE(prev != v.end()); + for (auto i = prev + 1; i != v.end(); prev = i, ++i) { + REQUIRE(*prev < *i); + } + }; + REQUIRE(v.size() == 4); + checkSorted(); + + SECTION("adding item puts it in the right position") { + v.insert(3); + REQUIRE(v.size() == 5); + REQUIRE(v.find(3) != v.end()); + checkSorted(); + } +} diff --git a/Telegram/SourceFiles/base/tests_main.cpp b/Telegram/SourceFiles/base/tests_main.cpp new file mode 100644 index 000000000..0696e477e --- /dev/null +++ b/Telegram/SourceFiles/base/tests_main.cpp @@ -0,0 +1,97 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#define CATCH_CONFIG_RUNNER +#include "catch.hpp" +#include "reporters/catch_reporter_compact.hpp" +#include <QFile> + +namespace Catch { + + struct MinimalReporter : CompactReporter { + MinimalReporter( ReporterConfig const& _config ) + : CompactReporter( _config ) + {} + + virtual void testRunEnded( TestRunStats const& _testRunStats ) { + printTotals( _testRunStats.totals ); + } + + private: + // Colour, message variants: + // - white: No tests ran. + // - red: Failed [both/all] N test cases, failed [both/all] M assertions. + // - white: Passed [both/all] N test cases (no assertions). + // - red: Failed N tests cases, failed M assertions. + // - green: Passed [both/all] N tests cases with M assertions. + + std::string bothOrAll( std::size_t count ) const { + return count == 1 ? std::string() : count == 2 ? "both " : "all " ; + } + + void printTotals( const Totals& totals ) const { + if( totals.testCases.total() == 0 ) { + } + else if( totals.testCases.failed == totals.testCases.total() ) { + Colour colour( Colour::ResultError ); + const std::string qualify_assertions_failed = + totals.assertions.failed == totals.assertions.total() ? + bothOrAll( totals.assertions.failed ) : std::string(); + stream << + "Failed " << bothOrAll( totals.testCases.failed ) + << pluralise( totals.testCases.failed, "test case" ) << ", " + "failed " << qualify_assertions_failed << + pluralise( totals.assertions.failed, "assertion" ) << '.'; + } + else if( totals.assertions.total() == 0 ) { + stream << + "Passed " << bothOrAll( totals.testCases.total() ) + << pluralise( totals.testCases.total(), "test case" ) + << " (no assertions)."; + } + else if( totals.assertions.failed ) { + Colour colour( Colour::ResultError ); + stream << + "Failed " << pluralise( totals.testCases.failed, "test case" ) << ", " + "failed " << pluralise( totals.assertions.failed, "assertion" ) << '.'; + } + else { + } + } + }; + + INTERNAL_CATCH_REGISTER_REPORTER( "minimal", MinimalReporter ) + +} // end namespace Catch + +int main(int argc, const char *argv[]) { + const char *catch_argv[] = { argv[0], "-r", "minimal" }; + constexpr auto catch_argc = sizeof(catch_argv) / sizeof(catch_argv[0]); + auto result = Catch::Session().run(catch_argc, catch_argv); + if (result == 0) { + for (auto i = 0; i != argc; ++i) { + if (argv[i] == QString("--touch") && i + 1 != argc) { + QFile(QFile::decodeName(argv[++i])).open(QIODevice::WriteOnly); + } + } + } + return (result < 0xff ? result : 0xff); +} + diff --git a/Telegram/ThirdParty/Catch b/Telegram/ThirdParty/Catch new file mode 160000 index 000000000..5ca44b687 --- /dev/null +++ b/Telegram/ThirdParty/Catch @@ -0,0 +1 @@ +Subproject commit 5ca44b68721833ae3731802ed99af67c6f38a53a diff --git a/Telegram/create.bat b/Telegram/create.bat index 9dcf9ac21..570e75f2b 100644 --- a/Telegram/create.bat +++ b/Telegram/create.bat @@ -4,7 +4,10 @@ set "FullScriptPath=%~dp0" set "FullExecPath=%cd%" set "Command=%1" -if "%Command%" == "header" ( +if "%Command%" == "test" ( + call :write_test %2 + exit /b %errorlevel% +) else if "%Command%" == "header" ( call :write_header %2 exit /b %errorlevel% ) else if "%Command%" == "source" ( @@ -87,6 +90,34 @@ exit /b %errorlevel% exit /b ) +:write_test +( + set "CommandPath=%1" + set "CommandPathUnix=!CommandPath:\=/!" + set "CommandPathWin=!CommandPath:/=\!" + + if "!CommandPathUnix!" == "" ( + echo Provide source path. + exit /b 1 + ) else if exist "SourceFiles\!CommandPathWin!.cpp" ( + echo This source already exists. + exit /b 1 + ) + echo Generating test !CommandPathUnix!.cpp.. + mkdir "SourceFiles\!CommandPathWin!.cpp" + rmdir "SourceFiles\!CommandPathWin!.cpp" + + call :write_comment !CommandPathWin!.cpp + set "quote=""" + set "quote=!quote:~0,1!" + set "source1=#include !quote!catch.hpp!quote!" + ( + echo !source1! + echo. + )>> "SourceFiles\!CommandPathWin!.cpp" + exit /b +) + :write_comment ( set "Path=%1" diff --git a/Telegram/gyp/Telegram.gyp b/Telegram/gyp/Telegram.gyp index 9d92b817d..0d1f38aa7 100644 --- a/Telegram/gyp/Telegram.gyp +++ b/Telegram/gyp/Telegram.gyp @@ -79,6 +79,7 @@ 'codegen.gyp:codegen_lang', 'codegen.gyp:codegen_numbers', 'codegen.gyp:codegen_style', + 'tests/tests.gyp:tests', 'utils.gyp:Updater', '../ThirdParty/libtgvoip/libtgvoip.gyp:libtgvoip', ], diff --git a/Telegram/gyp/qt.gypi b/Telegram/gyp/qt.gypi index 81c13a182..6032d0664 100644 --- a/Telegram/gyp/qt.gypi +++ b/Telegram/gyp/qt.gypi @@ -225,14 +225,14 @@ '<(linux_lib_ssl)', '<(linux_lib_crypto)', '<!@(python -c "for s in \'<(linux_lib_icu)\'.split(\' \'): print(s)")', - 'xcb', - 'X11', - 'X11-xcb', - 'dbus-1', - 'dl', - 'gthread-2.0', - 'glib-2.0', - 'pthread', + '-lxcb', + '-lX11', + '-lX11-xcb', + '-ldbus-1', + '-ldl', + '-lgthread-2.0', + '-lglib-2.0', + '-lpthread', ], 'include_dirs': [ '<(qt_loc)/mkspecs/linux-g++', diff --git a/Telegram/gyp/tests/common_test.gypi b/Telegram/gyp/tests/common_test.gypi new file mode 100644 index 000000000..1b91be262 --- /dev/null +++ b/Telegram/gyp/tests/common_test.gypi @@ -0,0 +1,34 @@ +# This file is part of Telegram Desktop, +# the official desktop version of Telegram messaging app, see https://telegram.org +# +# Telegram Desktop is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# It is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# In addition, as a special exception, the copyright holders give permission +# to link the code of portions of this program with the OpenSSL library. +# +# Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +# Copyright (c) 2014 John Preston, https://desktop.telegram.org + +{ + 'includes': [ + '../common_executable.gypi', + '../qt.gypi', + ], + 'include_dirs': [ + '<(src_loc)', + '<(submodules_loc)/GSL/include', + '<(submodules_loc)/variant/include', + '<(submodules_loc)/Catch/include' + ], + 'sources': [ + '<(src_loc)/base/tests_main.cpp', + ], +} diff --git a/Telegram/gyp/tests/list_tests.py b/Telegram/gyp/tests/list_tests.py new file mode 100644 index 000000000..9850b6972 --- /dev/null +++ b/Telegram/gyp/tests/list_tests.py @@ -0,0 +1,79 @@ +''' +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014 John Preston, https://desktop.telegram.org +''' +from __future__ import print_function +import sys +import os +import re +import time +import codecs + +def eprint(*args, **kwargs): + print(*args, file=sys.stderr, **kwargs) + sys.exit(1) + +my_path = os.path.dirname(os.path.realpath(__file__)).replace('\\', '/') + +next_input_path = 0 +input_path = '' +write_sources = 0 +next_self = 1 +for arg in sys.argv: + if next_self != 0: + next_self = 0 + continue + + if arg == '--sources': + write_sources = 1 + continue + + if arg == '--input': + next_input_path = 1 + continue + elif next_input_path == 1: + next_input_path = 0 + input_path = arg + continue + +tests_names = [] + +if input_path != '': + if not os.path.isfile(input_path): + eprint('Input path not found.') + else: + with open(input_path, 'r') as f: + for line in f: + test_name = line.strip() + if test_name[0:2] != '//' and test_name != '': + tests_names.append(test_name) + +if write_sources != 0: + tests_path = my_path + '/'; + if not os.path.isdir(tests_path): + os.mkdir(tests_path) + for test_name in tests_names: + test_path = tests_path + test_name + '.test' + if not os.path.isfile(test_path): + with open(test_path, 'w') as out: + out.write('1') + print(test_name + '.test') +else: + for test_name in tests_names: + print(test_name) diff --git a/Telegram/gyp/tests/tests.gyp b/Telegram/gyp/tests/tests.gyp new file mode 100644 index 000000000..105554be9 --- /dev/null +++ b/Telegram/gyp/tests/tests.gyp @@ -0,0 +1,77 @@ +# This file is part of Telegram Desktop, +# the official desktop version of Telegram messaging app, see https://telegram.org +# +# Telegram Desktop is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# It is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# In addition, as a special exception, the copyright holders give permission +# to link the code of portions of this program with the OpenSSL library. +# +# Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +# Copyright (c) 2014 John Preston, https://desktop.telegram.org + +{ + 'includes': [ + '../common.gypi', + ], + 'variables': { + 'libs_loc': '../../../../Libraries', + 'src_loc': '../../SourceFiles', + 'submodules_loc': '../../ThirdParty', + 'mac_target': '10.10', + 'list_tests_command': 'python <(DEPTH)/tests/list_tests.py --input <(DEPTH)/tests/tests_list.txt', + }, + 'targets': [{ + 'target_name': 'tests', + 'type': 'none', + 'includes': [ + '../common.gypi', + ], + 'dependencies': [ + '<!@(<(list_tests_command))', + ], + 'sources': [ + '<!@(<(list_tests_command) --sources)', + ], + 'rules': [{ + 'rule_name': 'run_tests', + 'extension': 'test', + 'inputs': [ + '<(PRODUCT_DIR)/<(RULE_INPUT_ROOT)<(exe_ext)', + ], + 'outputs': [ + '<(SHARED_INTERMEDIATE_DIR)/<(RULE_INPUT_ROOT).timestamp', + ], + 'action': [ + '<(PRODUCT_DIR)/<(RULE_INPUT_ROOT)<(exe_ext)', + '--touch', '<(SHARED_INTERMEDIATE_DIR)/<(RULE_INPUT_ROOT).timestamp', + ], + 'message': 'Running <(RULE_INPUT_ROOT)..', + }] + }, { + 'target_name': 'tests_flat_map', + 'includes': [ + 'common_test.gypi', + ], + 'sources': [ + '<(src_loc)/base/flat_map.h', + '<(src_loc)/base/flat_map_tests.cpp', + ], + }, { + 'target_name': 'tests_flat_set', + 'includes': [ + 'common_test.gypi', + ], + 'sources': [ + '<(src_loc)/base/flat_set.h', + '<(src_loc)/base/flat_set_tests.cpp', + ], + }], +} diff --git a/Telegram/gyp/tests/tests_list.txt b/Telegram/gyp/tests/tests_list.txt new file mode 100644 index 000000000..6d79c8e1e --- /dev/null +++ b/Telegram/gyp/tests/tests_list.txt @@ -0,0 +1,2 @@ +tests_flat_map +tests_flat_set