From f049836d686f5e7817682e700c3f4822fa03c9e7 Mon Sep 17 00:00:00 2001 From: Axel Huebl Date: Fri, 2 Nov 2018 10:15:27 +0100 Subject: [PATCH 001/220] CMake: Optional Install if Embedded When adding this library as embedded library with private "target link", e.g. only used inside private source files, the library does not need to be installed when the main project gets installed. This adds an additional option `JSON_Install` similar to the test-build control switch in order to skip installing headers and CMake config files if requested. Avoids using ```cmake add_subdirectory(path/to/nlohmann_json EXCLUDE_FROM_ALL) ``` which has further side-effects: https://cmake.org/cmake/help/v3.0/command/add_subdirectory.html --- CMakeLists.txt | 57 ++++++++++++++++++++++++++------------------------ README.md | 4 ++++ 2 files changed, 34 insertions(+), 27 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 230cbff8..716abf9d 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -16,6 +16,7 @@ include(ExternalProject) ## OPTIONS ## option(JSON_BuildTests "Build the unit tests when BUILD_TESTING is enabled." ON) +option(JSON_Install "Install CMake targets during install step." ON) option(JSON_MultipleHeaders "Use non-amalgamated version of the library." OFF) ## @@ -92,32 +93,34 @@ configure_file( @ONLY ) -install( - DIRECTORY ${NLOHMANN_JSON_INCLUDE_BUILD_DIR} - DESTINATION ${NLOHMANN_JSON_INCLUDE_INSTALL_DIR} -) -install( - FILES ${NLOHMANN_JSON_CMAKE_PROJECT_CONFIG_FILE} ${NLOHMANN_JSON_CMAKE_VERSION_CONFIG_FILE} - DESTINATION ${NLOHMANN_JSON_CONFIG_INSTALL_DIR} -) -if (NLOHMANN_ADD_NATVIS) +if(JSON_Install) install( - FILES ${NLOHMANN_NATVIS_FILE} - DESTINATION . -) + DIRECTORY ${NLOHMANN_JSON_INCLUDE_BUILD_DIR} + DESTINATION ${NLOHMANN_JSON_INCLUDE_INSTALL_DIR} + ) + install( + FILES ${NLOHMANN_JSON_CMAKE_PROJECT_CONFIG_FILE} ${NLOHMANN_JSON_CMAKE_VERSION_CONFIG_FILE} + DESTINATION ${NLOHMANN_JSON_CONFIG_INSTALL_DIR} + ) + if (NLOHMANN_ADD_NATVIS) + install( + FILES ${NLOHMANN_NATVIS_FILE} + DESTINATION . + ) + endif() + export( + TARGETS ${NLOHMANN_JSON_TARGET_NAME} + NAMESPACE ${PROJECT_NAME}:: + FILE ${NLOHMANN_JSON_CMAKE_PROJECT_TARGETS_FILE} + ) + install( + TARGETS ${NLOHMANN_JSON_TARGET_NAME} + EXPORT ${NLOHMANN_JSON_TARGETS_EXPORT_NAME} + INCLUDES DESTINATION ${NLOHMANN_JSON_INCLUDE_INSTALL_DIR} + ) + install( + EXPORT ${NLOHMANN_JSON_TARGETS_EXPORT_NAME} + NAMESPACE ${PROJECT_NAME}:: + DESTINATION ${NLOHMANN_JSON_CONFIG_INSTALL_DIR} + ) endif() -export( - TARGETS ${NLOHMANN_JSON_TARGET_NAME} - NAMESPACE ${PROJECT_NAME}:: - FILE ${NLOHMANN_JSON_CMAKE_PROJECT_TARGETS_FILE} -) -install( - TARGETS ${NLOHMANN_JSON_TARGET_NAME} - EXPORT ${NLOHMANN_JSON_TARGETS_EXPORT_NAME} - INCLUDES DESTINATION ${NLOHMANN_JSON_INCLUDE_INSTALL_DIR} -) -install( - EXPORT ${NLOHMANN_JSON_TARGETS_EXPORT_NAME} - NAMESPACE ${PROJECT_NAME}:: - DESTINATION ${NLOHMANN_JSON_CONFIG_INSTALL_DIR} -) diff --git a/README.md b/README.md index 1154a62e..818a1269 100644 --- a/README.md +++ b/README.md @@ -97,6 +97,10 @@ To embed the library directly into an existing CMake project, place the entire s # run from your own project's code. set(JSON_BuildTests OFF CACHE INTERNAL "") +# If you only include this third party in PRIVATE source files, you do not +# need to install it when your main project gets installed. +# set(JSON_Install OFF CACHE INTERNAL "") + # Don't use include(nlohmann_json/CMakeLists.txt) since that carries with it # inintended consequences that will break the build. It's generally # discouraged (although not necessarily well documented as such) to use From 2f44ac1def9899c6ac4b09e67fb313c36d172b96 Mon Sep 17 00:00:00 2001 From: onqtam Date: Sun, 13 Jan 2019 18:41:21 +0200 Subject: [PATCH 002/220] moved from Catch to doctest for unit tests --- .github/CONTRIBUTING.md | 2 +- Makefile | 2 - README.md | 2 +- test/CMakeLists.txt | 26 +- test/Makefile | 8 +- test/src/unit-algorithms.cpp | 2 +- test/src/unit-allocator.cpp | 25 +- test/src/unit-alt-string.cpp | 3 +- test/src/unit-bson.cpp | 18 +- test/src/unit-capacity.cpp | 2 +- test/src/unit-cbor.cpp | 29 +- test/src/unit-class_const_iterator.cpp | 3 +- test/src/unit-class_iterator.cpp | 3 +- test/src/unit-class_lexer.cpp | 3 +- test/src/unit-class_parser.cpp | 23 +- test/src/unit-comparison.cpp | 2 +- test/src/unit-concepts.cpp | 2 +- test/src/unit-constructor1.cpp | 4 +- test/src/unit-constructor2.cpp | 2 +- test/src/unit-convenience.cpp | 5 +- test/src/unit-conversions.cpp | 4 +- test/src/unit-deserialization.cpp | 3 +- test/src/unit-element_access1.cpp | 2 +- test/src/unit-element_access2.cpp | 6 +- test/src/unit-inspection.cpp | 6 +- test/src/unit-items.cpp | 2 +- test/src/unit-iterators1.cpp | 3 +- test/src/unit-iterators2.cpp | 2 +- test/src/unit-json_patch.cpp | 10 +- test/src/unit-json_pointer.cpp | 3 +- test/src/unit-merge_patch.cpp | 2 +- test/src/unit-meta.cpp | 2 +- test/src/unit-modifiers.cpp | 2 +- test/src/unit-msgpack.cpp | 17 +- test/src/unit-noexcept.cpp | 3 +- test/src/unit-pointer_access.cpp | 2 +- test/src/unit-readme.cpp | 7 +- test/src/unit-reference_access.cpp | 2 +- test/src/unit-regression.cpp | 22 +- test/src/unit-serialization.cpp | 5 +- test/src/unit-testsuites.cpp | 6 +- test/src/unit-to_chars.cpp | 2 +- test/src/unit-ubjson.cpp | 21 +- test/src/unit-udt.cpp | 17 +- test/src/unit-unicode.cpp | 10 +- test/src/unit-wstring.cpp | 3 +- test/src/unit.cpp | 4 +- test/thirdparty/catch/LICENSE_1_0.txt | 23 - test/thirdparty/catch/catch.hpp | 11689 ---------------- test/thirdparty/doctest/LICENSE.txt | 21 + test/thirdparty/doctest/doctest.h | 5269 +++++++ .../doctest/doctest_compatibility.h | 35 + 52 files changed, 5517 insertions(+), 11854 deletions(-) delete mode 100755 test/thirdparty/catch/LICENSE_1_0.txt delete mode 100644 test/thirdparty/catch/catch.hpp create mode 100644 test/thirdparty/doctest/LICENSE.txt create mode 100644 test/thirdparty/doctest/doctest.h create mode 100644 test/thirdparty/doctest/doctest_compatibility.h diff --git a/.github/CONTRIBUTING.md b/.github/CONTRIBUTING.md index 4ae0cb2d..f274b10d 100644 --- a/.github/CONTRIBUTING.md +++ b/.github/CONTRIBUTING.md @@ -32,7 +32,7 @@ To make changes, you need to edit the following files: 1. [`include/nlohmann/*`](https://github.com/nlohmann/json/tree/develop/include/nlohmann) - These files are the sources of the library. Before testing or creating a pull request, execute `make amalgamate` to regenerate `single_include/nlohmann/json.hpp`. -2. [`test/src/unit-*.cpp`](https://github.com/nlohmann/json/tree/develop/test/src) - These files contain the [Catch](https://github.com/philsquared/Catch) unit tests which currently cover [100 %](https://coveralls.io/github/nlohmann/json) of the library's code. +2. [`test/src/unit-*.cpp`](https://github.com/nlohmann/json/tree/develop/test/src) - These files contain the [doctest](https://github.com/onqtam/doctest) unit tests which currently cover [100 %](https://coveralls.io/github/nlohmann/json) of the library's code. If you add or change a feature, please also add a unit test to this file. The unit tests can be compiled and executed with diff --git a/Makefile b/Makefile index 297443a5..cb03ff80 100644 --- a/Makefile +++ b/Makefile @@ -113,7 +113,6 @@ doctest: # calling Clang with all warnings, except: # -Wno-documentation-unknown-command: code uses user-defined commands like @complexity -# -Wno-exit-time-destructors: warning in Catch code # -Wno-keyword-macro: unit-tests use "#define private public" # -Wno-deprecated-declarations: the library deprecated some functions # -Wno-weak-vtables: exception class is defined inline, but has virtual method @@ -128,7 +127,6 @@ pedantic_clang: -Werror \ -Weverything \ -Wno-documentation-unknown-command \ - -Wno-exit-time-destructors \ -Wno-keyword-macro \ -Wno-deprecated-declarations \ -Wno-weak-vtables \ diff --git a/README.md b/README.md index 7f041680..0c2bfca0 100644 --- a/README.md +++ b/README.md @@ -1240,13 +1240,13 @@ The library itself consists of a single header file licensed under the MIT licen - [**American fuzzy lop**](http://lcamtuf.coredump.cx/afl/) for fuzz testing - [**AppVeyor**](https://www.appveyor.com) for [continuous integration](https://ci.appveyor.com/project/nlohmann/json) on Windows - [**Artistic Style**](http://astyle.sourceforge.net) for automatic source code identation -- [**Catch**](https://github.com/philsquared/Catch) for the unit tests - [**Clang**](http://clang.llvm.org) for compilation with code sanitizers - [**CMake**](https://cmake.org) for build automation - [**Codacity**](https://www.codacy.com) for further [code analysis](https://www.codacy.com/app/nlohmann/json) - [**Coveralls**](https://coveralls.io) to measure [code coverage](https://coveralls.io/github/nlohmann/json) - [**Coverity Scan**](https://scan.coverity.com) for [static analysis](https://scan.coverity.com/projects/nlohmann-json) - [**cppcheck**](http://cppcheck.sourceforge.net) for static analysis +- [**doctest**](https://github.com/onqtam/doctest) for the unit tests - [**Doxygen**](http://www.stack.nl/~dimitri/doxygen/) to generate [documentation](https://nlohmann.github.io/json/) - [**git-update-ghpages**](https://github.com/rstacruz/git-update-ghpages) to upload the documentation to gh-pages - [**GitHub Changelog Generator**](https://github.com/skywinder/github-changelog-generator) to generate the [ChangeLog](https://github.com/nlohmann/json/blob/develop/ChangeLog.md) diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 04acdaab..f314ea66 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -22,7 +22,7 @@ if(JSON_NoExceptions) if(NOT MSVC) set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -DJSON_NOEXCEPTION") endif() - set(CATCH_TEST_FILTER -e) + set(DOCTEST_TEST_FILTER --no-throw) endif() if(JSON_Coverage) @@ -54,18 +54,18 @@ if(JSON_Coverage) endif() ############################################################################# -# Catch library with the main function to speed up build +# doctest library with the main function to speed up build ############################################################################# -add_library(catch_main OBJECT +add_library(doctest_main OBJECT "src/unit.cpp" ) -set_target_properties(catch_main PROPERTIES +set_target_properties(doctest_main PROPERTIES COMPILE_DEFINITIONS "$<$:_SCL_SECURE_NO_WARNINGS>" COMPILE_OPTIONS "$<$:/EHsc;$<$:/Od>>" ) -target_compile_features(catch_main PUBLIC cxx_std_11) -target_include_directories(catch_main PRIVATE "thirdparty/catch") +target_compile_features(doctest_main PUBLIC cxx_std_11) +target_include_directories(doctest_main PRIVATE "thirdparty/doctest") # https://stackoverflow.com/questions/2368811/how-to-set-warning-level-in-cmake if(MSVC) @@ -95,10 +95,10 @@ foreach(file ${files}) get_filename_component(file_basename ${file} NAME_WE) string(REGEX REPLACE "unit-([^$]+)" "test-\\1" testcase ${file_basename}) - add_executable(${testcase} $ ${file}) + add_executable(${testcase} $ ${file}) target_compile_definitions(${testcase} PRIVATE - CATCH_CONFIG_FAST_COMPILE - $<$:_SCL_SECURE_NO_WARNINGS> + DOCTEST_CONFIG_SUPER_FAST_ASSERTS + DOCTEST_THREAD_LOCAL ) target_compile_options(${testcase} PRIVATE $<$:/EHsc;$<$:/Od>> @@ -106,26 +106,26 @@ foreach(file ${files}) $<$:-Wno-deprecated-declarations> ) target_include_directories(${testcase} PRIVATE - thirdparty/catch + thirdparty/doctest thirdparty/fifo_map ) target_link_libraries(${testcase} ${NLOHMANN_JSON_TARGET_NAME}) add_test(NAME "${testcase}_default" - COMMAND ${testcase} ${CATCH_TEST_FILTER} + COMMAND ${testcase} ${DOCTEST_TEST_FILTER} WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} ) set_tests_properties("${testcase}_default" PROPERTIES LABELS "default") add_test(NAME "${testcase}_all" - COMMAND ${testcase} ${CATCH_TEST_FILTER} "*" + COMMAND ${testcase} ${DOCTEST_TEST_FILTER} --no-skip WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} ) set_tests_properties("${testcase}_all" PROPERTIES LABELS "all") if(JSON_Valgrind) add_test(NAME "${testcase}_valgrind" - COMMAND ${memcheck_command} ${CMAKE_CURRENT_BINARY_DIR}/${testcase} ${CATCH_TEST_FILTER} + COMMAND ${memcheck_command} ${CMAKE_CURRENT_BINARY_DIR}/${testcase} ${DOCTEST_TEST_FILTER} WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} ) set_tests_properties("${testcase}_valgrind" PROPERTIES LABELS "valgrind") diff --git a/test/Makefile b/test/Makefile index 4f00cbc7..499ff3c2 100644 --- a/test/Makefile +++ b/test/Makefile @@ -4,7 +4,7 @@ # additional flags CXXFLAGS += -std=c++11 -Wall -Wextra -pedantic -Wcast-align -Wcast-qual -Wno-ctor-dtor-privacy -Wdisabled-optimization -Wformat=2 -Winit-self -Wmissing-declarations -Wmissing-include-dirs -Wold-style-cast -Woverloaded-virtual -Wredundant-decls -Wshadow -Wsign-conversion -Wsign-promo -Wstrict-overflow=5 -Wswitch -Wundef -Wno-unused -Wnon-virtual-dtor -Wreorder -Wdeprecated -Wno-float-equal -CPPFLAGS += -I ../single_include -I . -I thirdparty/catch -I thirdparty/fifo_map -DCATCH_CONFIG_FAST_COMPILE +CPPFLAGS += -I ../single_include -I . -I thirdparty/doctest -I thirdparty/fifo_map -DDOCTEST_CONFIG_SUPER_FAST_ASSERTS SOURCES = src/unit.cpp \ src/unit-algorithms.cpp \ @@ -63,11 +63,11 @@ clean: # single test file ############################################################################## -json_unit: $(OBJECTS) ../single_include/nlohmann/json.hpp thirdparty/catch/catch.hpp +json_unit: $(OBJECTS) ../single_include/nlohmann/json.hpp thirdparty/doctest/doctest.h @echo "[CXXLD] $@" @$(CXX) $(CXXFLAGS) $(LDFLAGS) $(OBJECTS) -o $@ -%.o: %.cpp ../single_include/nlohmann/json.hpp thirdparty/catch/catch.hpp +%.o: %.cpp ../single_include/nlohmann/json.hpp thirdparty/doctest/doctest.h @echo "[CXX] $@" @$(CXX) $(CXXFLAGS) $(CPPFLAGS) -c $< -o $@ @@ -76,7 +76,7 @@ json_unit: $(OBJECTS) ../single_include/nlohmann/json.hpp thirdparty/catch/catch # individual test cases ############################################################################## -test-%: src/unit-%.o src/unit.o ../single_include/nlohmann/json.hpp thirdparty/catch/catch.hpp +test-%: src/unit-%.o src/unit.o ../single_include/nlohmann/json.hpp thirdparty/doctest/doctest.h @echo "[CXXLD] $@" @$(CXX) $(CXXFLAGS) $(CPPFLAGS) $(LDFLAGS) $< src/unit.o -o $@ diff --git a/test/src/unit-algorithms.cpp b/test/src/unit-algorithms.cpp index 547d2294..adcf141f 100644 --- a/test/src/unit-algorithms.cpp +++ b/test/src/unit-algorithms.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-allocator.cpp b/test/src/unit-allocator.cpp index 06d56d5e..1d9de260 100644 --- a/test/src/unit-allocator.cpp +++ b/test/src/unit-allocator.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private // special test case to check if memory is leaked if constructor throws @@ -60,7 +61,7 @@ TEST_CASE("bad_alloc") bad_allocator>; // creating an object should throw - CHECK_THROWS_AS(bad_json(bad_json::value_t::object), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = bad_json(bad_json::value_t::object), std::bad_alloc&); } } @@ -144,7 +145,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::object; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).object)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } SECTION("array") @@ -153,7 +154,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::array; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).array)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } SECTION("string") @@ -162,7 +163,7 @@ TEST_CASE("controlled bad_alloc") auto t = my_json::value_t::string; CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(t).string)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(t), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json::json_value(t), std::bad_alloc&); next_construct_fails = false; } } @@ -173,7 +174,7 @@ TEST_CASE("controlled bad_alloc") my_json::string_t v("foo"); CHECK_NOTHROW(my_allocator_clean_up(my_json::json_value(v).string)); next_construct_fails = true; - CHECK_THROWS_AS(my_json::json_value(v), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json::json_value(v), std::bad_alloc&); next_construct_fails = false; } } @@ -184,9 +185,9 @@ TEST_CASE("controlled bad_alloc") { next_construct_fails = false; std::map v {{"foo", "bar"}}; - CHECK_NOTHROW(my_json(v)); + CHECK_NOTHROW(auto tmp = my_json(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(v), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json(v), std::bad_alloc&); next_construct_fails = false; } @@ -194,9 +195,9 @@ TEST_CASE("controlled bad_alloc") { next_construct_fails = false; std::vector v {"foo", "bar", "baz"}; - CHECK_NOTHROW(my_json(v)); + CHECK_NOTHROW(auto tmp = my_json(v)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(v), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json(v), std::bad_alloc&); next_construct_fails = false; } @@ -213,9 +214,9 @@ TEST_CASE("controlled bad_alloc") { next_construct_fails = false; std::string s("foo"); - CHECK_NOTHROW(my_json(s)); + CHECK_NOTHROW(auto tmp = my_json(s)); next_construct_fails = true; - CHECK_THROWS_AS(my_json(s), std::bad_alloc&); + CHECK_THROWS_AS(auto tmp = my_json(s), std::bad_alloc&); next_construct_fails = false; } } diff --git a/test/src/unit-alt-string.cpp b/test/src/unit-alt-string.cpp index 9db580b4..23b40d89 100644 --- a/test/src/unit-alt-string.cpp +++ b/test/src/unit-alt-string.cpp @@ -27,9 +27,10 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include + #include #include diff --git a/test/src/unit-bson.cpp b/test/src/unit-bson.cpp index 7ae7db36..475f2755 100644 --- a/test/src/unit-bson.cpp +++ b/test/src/unit-bson.cpp @@ -27,11 +27,13 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; + #include +#include TEST_CASE("BSON") { @@ -1138,7 +1140,7 @@ TEST_CASE("BSON numerical data") }; CHECK_THROWS_AS(json::to_bson(j), json::out_of_range&); - CHECK_THROWS_WITH(json::to_bson(j), "[json.exception.out_of_range.407] integer number " + std::to_string(i) + " cannot be represented by BSON as it does not fit int64"); + CHECK_THROWS_WITH_STD_STR(json::to_bson(j), "[json.exception.out_of_range.407] integer number " + std::to_string(i) + " cannot be represented by BSON as it does not fit int64"); } } @@ -1146,7 +1148,7 @@ TEST_CASE("BSON numerical data") } } -TEST_CASE("BSON roundtrips", "[hide]") +TEST_CASE("BSON roundtrips" * doctest::skip()) { SECTION("reference files") { @@ -1161,8 +1163,8 @@ TEST_CASE("BSON roundtrips", "[hide]") { CAPTURE(filename) - SECTION(filename + ": std::vector") { + INFO_WITH_TEMP(filename + ": std::vector"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1179,8 +1181,8 @@ TEST_CASE("BSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": std::ifstream") { + INFO_WITH_TEMP(filename + ": std::ifstream"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1194,8 +1196,8 @@ TEST_CASE("BSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": uint8_t* and size") { + INFO_WITH_TEMP(filename + ": uint8_t* and size"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1212,8 +1214,8 @@ TEST_CASE("BSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": output to output adapters") { + INFO_WITH_TEMP(filename + ": output to output adapters"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1224,8 +1226,8 @@ TEST_CASE("BSON roundtrips", "[hide]") (std::istreambuf_iterator(f_bson)), std::istreambuf_iterator()); - SECTION(filename + ": output adapters: std::vector") { + INFO_WITH_TEMP(filename + ": output adapters: std::vector"); std::vector vec; json::to_bson(j1, vec); diff --git a/test/src/unit-capacity.cpp b/test/src/unit-capacity.cpp index a2332613..0ab187f9 100644 --- a/test/src/unit-capacity.cpp +++ b/test/src/unit-capacity.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-cbor.cpp b/test/src/unit-cbor.cpp index e96f4cd0..7cfdc383 100644 --- a/test/src/unit-cbor.cpp +++ b/test/src/unit-cbor.cpp @@ -27,12 +27,15 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; #include +#include +#include +#include class SaxCountdown { @@ -1586,7 +1589,8 @@ TEST_CASE("single CBOR roundtrip") } } -TEST_CASE("CBOR regressions", "[!throws]") +#if not defined(JSON_NOEXCEPTION) +TEST_CASE("CBOR regressions") { SECTION("fuzz test results") { @@ -1655,12 +1659,13 @@ TEST_CASE("CBOR regressions", "[!throws]") } } } +#endif -TEST_CASE("CBOR roundtrips", "[hide]") +TEST_CASE("CBOR roundtrips" * doctest::skip()) { SECTION("input from flynn") { - // most of these are exluded due to differences in key order (not a real problem) + // most of these are excluded due to differences in key order (not a real problem) auto exclude_packed = std::set { "test/data/json.org/1.json", @@ -1827,8 +1832,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") { CAPTURE(filename) - SECTION(filename + ": std::vector") { + INFO_WITH_TEMP(filename + ": std::vector"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1845,8 +1850,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": std::ifstream") { + INFO_WITH_TEMP(filename + ": std::ifstream"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1860,8 +1865,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": uint8_t* and size") { + INFO_WITH_TEMP(filename + ": uint8_t* and size"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1878,8 +1883,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": output to output adapters") { + INFO_WITH_TEMP(filename + ": output to output adapters"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1892,8 +1897,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") if (!exclude_packed.count(filename)) { - SECTION(filename + ": output adapters: std::vector") { + INFO_WITH_TEMP(filename + ": output adapters: std::vector"); std::vector vec; json::to_cbor(j1, vec); CHECK(vec == packed); @@ -1904,7 +1909,8 @@ TEST_CASE("CBOR roundtrips", "[hide]") } } -TEST_CASE("all CBOR first bytes", "[!throws]") +#if not defined(JSON_NOEXCEPTION) +TEST_CASE("all CBOR first bytes") { // these bytes will fail immediately with exception parse_error.112 std::set unsupported = @@ -1968,7 +1974,7 @@ TEST_CASE("all CBOR first bytes", "[!throws]") { // check that parse_error.112 is only thrown if the // first byte is in the unsupported set - CAPTURE(e.what()) + INFO_WITH_TEMP(e.what()); if (std::find(unsupported.begin(), unsupported.end(), byte) != unsupported.end()) { CHECK(e.id == 112); @@ -1980,6 +1986,7 @@ TEST_CASE("all CBOR first bytes", "[!throws]") } } } +#endif TEST_CASE("examples from RFC 7049 Appendix A") { diff --git a/test/src/unit-class_const_iterator.cpp b/test/src/unit-class_const_iterator.cpp index 40ce1dd5..caa2a001 100644 --- a/test/src/unit-class_const_iterator.cpp +++ b/test/src/unit-class_const_iterator.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private TEST_CASE("const_iterator class") { diff --git a/test/src/unit-class_iterator.cpp b/test/src/unit-class_iterator.cpp index b7d132e0..841aed3e 100644 --- a/test/src/unit-class_iterator.cpp +++ b/test/src/unit-class_iterator.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private TEST_CASE("iterator class") { diff --git a/test/src/unit-class_lexer.cpp b/test/src/unit-class_lexer.cpp index a2dff573..0200c86f 100644 --- a/test/src/unit-class_lexer.cpp +++ b/test/src/unit-class_lexer.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private // shortcut to scan a string literal json::lexer::token_type scan_string(const char* s); diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 2d2929ee..b3fcef22 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private #include @@ -1105,8 +1106,8 @@ TEST_CASE("parser class") // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parser_helper(s.c_str()), - "[json.exception.parse_error.101] parse error at line 1, column 3: syntax error while parsing value - invalid string: forbidden character after backslash; last read: '\"\\" + std::string(1, static_cast(c)) + "'"); + CHECK_THROWS_WITH_STD_STR(parser_helper(s.c_str()), + "[json.exception.parse_error.101] parse error at line 1, column 3: syntax error while parsing value - invalid string: forbidden character after backslash; last read: '\"\\" + std::string(1, static_cast(c)) + "'"); } break; } @@ -1181,8 +1182,8 @@ TEST_CASE("parser class") // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parser_helper(s1.c_str()), - "[json.exception.parse_error.101] parse error at line 1, column 7: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s1.substr(0, 7) + "'"); + CHECK_THROWS_WITH_STD_STR(parser_helper(s1.c_str()), + "[json.exception.parse_error.101] parse error at line 1, column 7: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s1.substr(0, 7) + "'"); } CAPTURE(s2) @@ -1190,8 +1191,8 @@ TEST_CASE("parser class") // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parser_helper(s2.c_str()), - "[json.exception.parse_error.101] parse error at line 1, column 6: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s2.substr(0, 6) + "'"); + CHECK_THROWS_WITH_STD_STR(parser_helper(s2.c_str()), + "[json.exception.parse_error.101] parse error at line 1, column 6: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s2.substr(0, 6) + "'"); } CAPTURE(s3) @@ -1199,8 +1200,8 @@ TEST_CASE("parser class") // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parser_helper(s3.c_str()), - "[json.exception.parse_error.101] parse error at line 1, column 5: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s3.substr(0, 5) + "'"); + CHECK_THROWS_WITH_STD_STR(parser_helper(s3.c_str()), + "[json.exception.parse_error.101] parse error at line 1, column 5: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s3.substr(0, 5) + "'"); } CAPTURE(s4) @@ -1208,8 +1209,8 @@ TEST_CASE("parser class") // only check error message if c is not a control character if (c > 0x1f) { - CHECK_THROWS_WITH(parser_helper(s4.c_str()), - "[json.exception.parse_error.101] parse error at line 1, column 4: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s4.substr(0, 4) + "'"); + CHECK_THROWS_WITH_STD_STR(parser_helper(s4.c_str()), + "[json.exception.parse_error.101] parse error at line 1, column 4: syntax error while parsing value - invalid string: '\\u' must be followed by 4 hex digits; last read: '" + s4.substr(0, 4) + "'"); } } } diff --git a/test/src/unit-comparison.cpp b/test/src/unit-comparison.cpp index e729d1f2..d030cea8 100644 --- a/test/src/unit-comparison.cpp +++ b/test/src/unit-comparison.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-concepts.cpp b/test/src/unit-concepts.cpp index 870ef52f..c73af279 100644 --- a/test/src/unit-concepts.cpp +++ b/test/src/unit-concepts.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-constructor1.cpp b/test/src/unit-constructor1.cpp index 205daeb3..65e50c5a 100644 --- a/test/src/unit-constructor1.cpp +++ b/test/src/unit-constructor1.cpp @@ -27,16 +27,18 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private #include #include #include #include +#include #include #include #include diff --git a/test/src/unit-constructor2.cpp b/test/src/unit-constructor2.cpp index 809b4551..39316ed0 100644 --- a/test/src/unit-constructor2.cpp +++ b/test/src/unit-constructor2.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-convenience.cpp b/test/src/unit-convenience.cpp index dda515f2..ea55338f 100644 --- a/test/src/unit-convenience.cpp +++ b/test/src/unit-convenience.cpp @@ -27,11 +27,14 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private + +#include void check_escaped(const char* original, const char* escaped = "", const bool ensure_ascii = false); void check_escaped(const char* original, const char* escaped, const bool ensure_ascii) diff --git a/test/src/unit-conversions.cpp b/test/src/unit-conversions.cpp index 368cb4f5..8dba9465 100644 --- a/test/src/unit-conversions.cpp +++ b/test/src/unit-conversions.cpp @@ -27,15 +27,17 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private #include #include #include +#include #include #include #include diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index f92241ff..b8163f73 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -27,12 +27,13 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; #include +#include #include struct SaxEventLogger : public nlohmann::json_sax diff --git a/test/src/unit-element_access1.cpp b/test/src/unit-element_access1.cpp index 171feef4..59df62ea 100644 --- a/test/src/unit-element_access1.cpp +++ b/test/src/unit-element_access1.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 52a30073..78958dfa 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; @@ -985,7 +985,8 @@ TEST_CASE("element access 2") } } -TEST_CASE("element access 2 (throwing tests)", "[!throws]") +#if not defined(JSON_NOEXCEPTION) +TEST_CASE("element access 2 (throwing tests)") { SECTION("object") { @@ -1018,3 +1019,4 @@ TEST_CASE("element access 2 (throwing tests)", "[!throws]") } } } +#endif diff --git a/test/src/unit-inspection.cpp b/test/src/unit-inspection.cpp index a529f098..1a31cff2 100644 --- a/test/src/unit-inspection.cpp +++ b/test/src/unit-inspection.cpp @@ -27,12 +27,14 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" -#include #include using nlohmann::json; +#include +#include + TEST_CASE("object inspection") { SECTION("convenience type checker") diff --git a/test/src/unit-items.cpp b/test/src/unit-items.cpp index c2f43cd4..9d77e5c9 100644 --- a/test/src/unit-items.cpp +++ b/test/src/unit-items.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-iterators1.cpp b/test/src/unit-iterators1.cpp index 1923cf60..d825e728 100644 --- a/test/src/unit-iterators1.cpp +++ b/test/src/unit-iterators1.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private TEST_CASE("iterators 1") { diff --git a/test/src/unit-iterators2.cpp b/test/src/unit-iterators2.cpp index e6536389..1f3cb9f9 100644 --- a/test/src/unit-iterators2.cpp +++ b/test/src/unit-iterators2.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-json_patch.cpp b/test/src/unit-json_patch.cpp index b72f1148..a24f1db5 100644 --- a/test/src/unit-json_patch.cpp +++ b/test/src/unit-json_patch.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; @@ -342,7 +342,7 @@ TEST_CASE("JSON patch") // check that evaluation throws CHECK_THROWS_AS(doc.patch(patch), json::other_error&); - CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); + CHECK_THROWS_WITH_STD_STR(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } SECTION("A.10. Adding a Nested Member Object") @@ -483,7 +483,7 @@ TEST_CASE("JSON patch") // check that evaluation throws CHECK_THROWS_AS(doc.patch(patch), json::other_error&); - CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); + CHECK_THROWS_WITH_STD_STR(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } SECTION("A.16. Adding an Array Value") @@ -1182,7 +1182,7 @@ TEST_CASE("JSON patch") // the test will fail CHECK_THROWS_AS(doc.patch(patch), json::other_error&); - CHECK_THROWS_WITH(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); + CHECK_THROWS_WITH_STD_STR(doc.patch(patch), "[json.exception.other_error.501] unsuccessful: " + patch[0].dump()); } } } @@ -1268,7 +1268,7 @@ TEST_CASE("JSON patch") for (const auto& test : suite) { - CAPTURE(test.value("comment", "")) + INFO_WITH_TEMP(test.value("comment", "")); // skip tests marked as disabled if (test.value("disabled", false)) diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index ac9cd71e..2f5ac4fd 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -27,11 +27,12 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #define private public #include using nlohmann::json; +#undef private TEST_CASE("JSON pointers") { diff --git a/test/src/unit-merge_patch.cpp b/test/src/unit-merge_patch.cpp index 901d3568..3a557c6d 100644 --- a/test/src/unit-merge_patch.cpp +++ b/test/src/unit-merge_patch.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-meta.cpp b/test/src/unit-meta.cpp index 46f2a9aa..28eee10b 100644 --- a/test/src/unit-meta.cpp +++ b/test/src/unit-meta.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-modifiers.cpp b/test/src/unit-modifiers.cpp index 9e53deab..57867237 100644 --- a/test/src/unit-modifiers.cpp +++ b/test/src/unit-modifiers.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-msgpack.cpp b/test/src/unit-msgpack.cpp index 544ed224..2d6a6974 100644 --- a/test/src/unit-msgpack.cpp +++ b/test/src/unit-msgpack.cpp @@ -27,12 +27,15 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; #include +#include +#include +#include class SaxCountdown { @@ -1345,7 +1348,7 @@ TEST_CASE("single MessagePack roundtrip") } -TEST_CASE("MessagePack roundtrips", "[hide]") +TEST_CASE("MessagePack roundtrips" * doctest::skip()) { SECTION("input from msgpack-python") { @@ -1519,8 +1522,8 @@ TEST_CASE("MessagePack roundtrips", "[hide]") { CAPTURE(filename) - SECTION(filename + ": std::vector") { + INFO_WITH_TEMP(filename + ": std::vector"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1537,8 +1540,8 @@ TEST_CASE("MessagePack roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": std::ifstream") { + INFO_WITH_TEMP(filename + ": std::ifstream"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1552,8 +1555,8 @@ TEST_CASE("MessagePack roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": uint8_t* and size") { + INFO_WITH_TEMP(filename + ": uint8_t* and size"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1570,8 +1573,8 @@ TEST_CASE("MessagePack roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": output to output adapters") { + INFO_WITH_TEMP(filename + ": output to output adapters"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -1584,8 +1587,8 @@ TEST_CASE("MessagePack roundtrips", "[hide]") if (!exclude_packed.count(filename)) { - SECTION(filename + ": output adapters: std::vector") { + INFO_WITH_TEMP(filename + ": output adapters: std::vector"); std::vector vec; json::to_msgpack(j1, vec); CHECK(vec == packed); diff --git a/test/src/unit-noexcept.cpp b/test/src/unit-noexcept.cpp index 3e1e544d..0aeda12a 100644 --- a/test/src/unit-noexcept.cpp +++ b/test/src/unit-noexcept.cpp @@ -27,7 +27,8 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" + #include using nlohmann::json; diff --git a/test/src/unit-pointer_access.cpp b/test/src/unit-pointer_access.cpp index ab409282..81025aec 100644 --- a/test/src/unit-pointer_access.cpp +++ b/test/src/unit-pointer_access.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-readme.cpp b/test/src/unit-readme.cpp index 280a834e..ffddaf36 100644 --- a/test/src/unit-readme.cpp +++ b/test/src/unit-readme.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; @@ -35,16 +35,19 @@ using nlohmann::json; #include #include #include +#include #include #include #include +#include +#include #if defined(_MSC_VER) #pragma warning (push) #pragma warning (disable : 4189) // local variable is initialized but not referenced #endif -TEST_CASE("README", "[hide]") +TEST_CASE("README" * doctest::skip()) { { // redirect std::cout for the README file diff --git a/test/src/unit-reference_access.cpp b/test/src/unit-reference_access.cpp index bc54ef52..907ed033 100644 --- a/test/src/unit-reference_access.cpp +++ b/test/src/unit-reference_access.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index e739e3c3..b157537e 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -27,11 +27,20 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" + +// for some reason including this after the json header leads to linker errors with VS 2017... +#include #define private public #include using nlohmann::json; +#undef private + +#include +#include +#include +#include #if (defined(__cplusplus) && __cplusplus >= 201703L) || (defined(_HAS_CXX17) && _HAS_CXX17 == 1) // fix for issue #464 #define JSON_HAS_CPP_17 @@ -43,10 +52,6 @@ using nlohmann::json; #include "fifo_map.hpp" -#include -#include -#include - ///////////////////////////////////////////////////////////////////// // for #972 ///////////////////////////////////////////////////////////////////// @@ -295,8 +300,7 @@ TEST_CASE("regression tests") nlohmann::basic_json; custom_json j; j["int_1"] = 1; - // we need to cast to int to compile with Catch - the value is int32_t - CHECK(static_cast(j["int_1"]) == 1); + CHECK(j["int_1"] == 1); // tests for correct handling of non-standard integers that overflow the type selected by the user @@ -1710,7 +1714,8 @@ TEST_CASE("regression tests") } } -TEST_CASE("regression tests, exceptions dependent", "[!throws]") +#if not defined(JSON_NOEXCEPTION) +TEST_CASE("regression tests, exceptions dependent") { SECTION("issue #1340 - eof not set on exhausted input stream") { @@ -1722,3 +1727,4 @@ TEST_CASE("regression tests, exceptions dependent", "[!throws]") CHECK(s.eof()); } } +#endif diff --git a/test/src/unit-serialization.cpp b/test/src/unit-serialization.cpp index 51ee9b57..ce7fd8de 100644 --- a/test/src/unit-serialization.cpp +++ b/test/src/unit-serialization.cpp @@ -27,11 +27,14 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; +#include +#include + TEST_CASE("serialization") { SECTION("operator<<") diff --git a/test/src/unit-testsuites.cpp b/test/src/unit-testsuites.cpp index f930c974..aecc21e0 100644 --- a/test/src/unit-testsuites.cpp +++ b/test/src/unit-testsuites.cpp @@ -27,7 +27,7 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; @@ -457,7 +457,7 @@ TEST_CASE("RFC 7159 examples") } )"; - CHECK_NOTHROW(json(json_contents)); + CHECK_NOTHROW(auto tmp = json(json_contents)); } { @@ -484,7 +484,7 @@ TEST_CASE("RFC 7159 examples") "Country": "US" } ])"; - CHECK_NOTHROW(json(json_contents)); + CHECK_NOTHROW(auto tmp = json(json_contents)); } CHECK(json::parse("\"Hello world!\"") == json("Hello world!")); diff --git a/test/src/unit-to_chars.cpp b/test/src/unit-to_chars.cpp index a8c7492c..17f1ead0 100644 --- a/test/src/unit-to_chars.cpp +++ b/test/src/unit-to_chars.cpp @@ -31,7 +31,7 @@ SOFTWARE. // Only compile these tests if 'float' and 'double' are IEEE-754 single- and // double-precision numbers, resp. -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::detail::dtoa_impl::reinterpret_bits; diff --git a/test/src/unit-ubjson.cpp b/test/src/unit-ubjson.cpp index 6279a5d3..1e42a7af 100644 --- a/test/src/unit-ubjson.cpp +++ b/test/src/unit-ubjson.cpp @@ -27,12 +27,13 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include using nlohmann::json; #include +#include class SaxCountdown { @@ -2113,7 +2114,8 @@ TEST_CASE("Universal Binary JSON Specification Examples 1") } } -TEST_CASE("all UBJSON first bytes", "[!throws]") +#if not defined(JSON_NOEXCEPTION) +TEST_CASE("all UBJSON first bytes") { // these bytes will fail immediately with exception parse_error.112 std::set supported = @@ -2134,7 +2136,7 @@ TEST_CASE("all UBJSON first bytes", "[!throws]") { // check that parse_error.112 is only thrown if the // first byte is not in the supported set - CAPTURE(e.what()) + INFO_WITH_TEMP(e.what()); if (std::find(supported.begin(), supported.end(), byte) == supported.end()) { CHECK(e.id == 112); @@ -2146,8 +2148,9 @@ TEST_CASE("all UBJSON first bytes", "[!throws]") } } } +#endif -TEST_CASE("UBJSON roundtrips", "[hide]") +TEST_CASE("UBJSON roundtrips" * doctest::skip()) { SECTION("input from self-generated UBJSON files") { @@ -2199,8 +2202,8 @@ TEST_CASE("UBJSON roundtrips", "[hide]") { CAPTURE(filename) - SECTION(filename + ": std::vector") { + INFO_WITH_TEMP(filename + ": std::vector"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -2217,8 +2220,8 @@ TEST_CASE("UBJSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": std::ifstream") { + INFO_WITH_TEMP(filename + ": std::ifstream"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -2232,8 +2235,8 @@ TEST_CASE("UBJSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": uint8_t* and size") { + INFO_WITH_TEMP(filename + ": uint8_t* and size"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -2250,8 +2253,8 @@ TEST_CASE("UBJSON roundtrips", "[hide]") CHECK(j1 == j2); } - SECTION(filename + ": output to output adapters") { + INFO_WITH_TEMP(filename + ": output to output adapters"); // parse JSON file std::ifstream f_json(filename); json j1 = json::parse(f_json); @@ -2262,8 +2265,8 @@ TEST_CASE("UBJSON roundtrips", "[hide]") (std::istreambuf_iterator(f_ubjson)), std::istreambuf_iterator()); - SECTION(filename + ": output adapters: std::vector") { + INFO_WITH_TEMP(filename + ": output adapters: std::vector"); std::vector vec; json::to_ubjson(j1, vec); CHECK(vec == packed); diff --git a/test/src/unit-udt.cpp b/test/src/unit-udt.cpp index 833a4c71..bb4b3ab3 100644 --- a/test/src/unit-udt.cpp +++ b/test/src/unit-udt.cpp @@ -27,10 +27,9 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include - using nlohmann::json; #include @@ -236,7 +235,7 @@ void from_json(const nlohmann::json& j, contact_book& cb) } } -TEST_CASE("basic usage", "[udt]") +TEST_CASE("basic usage" * doctest::test_suite("udt")) { // a bit narcissic maybe :) ? @@ -392,7 +391,7 @@ struct adl_serializer }; } -TEST_CASE("adl_serializer specialization", "[udt]") +TEST_CASE("adl_serializer specialization" * doctest::test_suite("udt")) { SECTION("partial specialization") { @@ -468,7 +467,7 @@ struct adl_serializer> }; } -TEST_CASE("even supported types can be specialized", "[udt]") +TEST_CASE("even supported types can be specialized" * doctest::test_suite("udt")) { json j = std::vector {1.0, 2.0, 3.0}; CHECK(j.dump() == R"("hijacked!")"); @@ -509,7 +508,7 @@ struct adl_serializer> }; } -TEST_CASE("Non-copyable types", "[udt]") +TEST_CASE("Non-copyable types" * doctest::test_suite("udt")) { SECTION("to_json") { @@ -651,7 +650,7 @@ std::ostream& operator<<(std::ostream& os, small_pod l) } } -TEST_CASE("custom serializer for pods", "[udt]") +TEST_CASE("custom serializer for pods" * doctest::test_suite("udt")) { using custom_json = nlohmann::basic_json struct is_constructible_patched())))> : std::true_type {}; } -TEST_CASE("an incomplete type does not trigger a compiler error in non-evaluated context", "[udt]") +TEST_CASE("an incomplete type does not trigger a compiler error in non-evaluated context" * doctest::test_suite("udt")) { static_assert(not is_constructible_patched::value, ""); } diff --git a/test/src/unit-unicode.cpp b/test/src/unit-unicode.cpp index 57c3c505..aaa3b585 100644 --- a/test/src/unit-unicode.cpp +++ b/test/src/unit-unicode.cpp @@ -27,14 +27,20 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" + +// for some reason including this after the json header leads to linker errors with VS 2017... +#include #define private public #include using nlohmann::json; +#undef private #include +#include #include +#include extern size_t calls; size_t calls = 0; @@ -160,7 +166,7 @@ void check_utf8string(bool success_expected, int byte1, int byte2 = -1, int byte } } -TEST_CASE("Unicode", "[hide]") +TEST_CASE("Unicode" * doctest::skip()) { SECTION("RFC 3629") { diff --git a/test/src/unit-wstring.cpp b/test/src/unit-wstring.cpp index 8434b199..32bc4a98 100644 --- a/test/src/unit-wstring.cpp +++ b/test/src/unit-wstring.cpp @@ -27,10 +27,9 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#include "catch.hpp" +#include "doctest_compatibility.h" #include - using nlohmann::json; bool wstring_is_utf16(); diff --git a/test/src/unit.cpp b/test/src/unit.cpp index 8ea7e2e8..7bad90d7 100644 --- a/test/src/unit.cpp +++ b/test/src/unit.cpp @@ -27,5 +27,5 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#define CATCH_CONFIG_MAIN -#include "catch.hpp" +#define DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN +#include "doctest_compatibility.h" diff --git a/test/thirdparty/catch/LICENSE_1_0.txt b/test/thirdparty/catch/LICENSE_1_0.txt deleted file mode 100755 index 36b7cd93..00000000 --- a/test/thirdparty/catch/LICENSE_1_0.txt +++ /dev/null @@ -1,23 +0,0 @@ -Boost Software License - Version 1.0 - August 17th, 2003 - -Permission is hereby granted, free of charge, to any person or organization -obtaining a copy of the software and accompanying documentation covered by -this license (the "Software") to use, reproduce, display, distribute, -execute, and transmit the Software, and to prepare derivative works of the -Software, and to permit third-parties to whom the Software is furnished to -do so, all subject to the following: - -The copyright notices in the Software and this entire statement, including -the above license grant, this restriction and the following disclaimer, -must be included in all copies of the Software, in whole or in part, and -all derivative works of the Software, unless such copies or derivative -works are solely in the form of machine-executable object code generated by -a source language processor. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT -SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE -FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, -ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER -DEALINGS IN THE SOFTWARE. diff --git a/test/thirdparty/catch/catch.hpp b/test/thirdparty/catch/catch.hpp deleted file mode 100644 index fdb046fe..00000000 --- a/test/thirdparty/catch/catch.hpp +++ /dev/null @@ -1,11689 +0,0 @@ -/* - * Catch v1.12.2 - * Generated: 2018-05-14 15:10:01.112442 - * ---------------------------------------------------------- - * This file has been merged from multiple headers. Please don't edit it directly - * Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. - * - * 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) - */ -#ifndef TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED -#define TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED - -#define TWOBLUECUBES_CATCH_HPP_INCLUDED - -#ifdef __clang__ -# pragma clang system_header -#elif defined __GNUC__ -# pragma GCC system_header -#endif - -// #included from: internal/catch_suppress_warnings.h - -#ifdef __clang__ -# ifdef __ICC // icpc defines the __clang__ macro -# pragma warning(push) -# pragma warning(disable: 161 1682) -# else // __ICC -# pragma clang diagnostic ignored "-Wglobal-constructors" -# pragma clang diagnostic ignored "-Wvariadic-macros" -# pragma clang diagnostic ignored "-Wc99-extensions" -# pragma clang diagnostic ignored "-Wunused-variable" -# pragma clang diagnostic push -# pragma clang diagnostic ignored "-Wpadded" -# pragma clang diagnostic ignored "-Wc++98-compat" -# pragma clang diagnostic ignored "-Wc++98-compat-pedantic" -# pragma clang diagnostic ignored "-Wswitch-enum" -# pragma clang diagnostic ignored "-Wcovered-switch-default" -# endif -#elif defined __GNUC__ -# pragma GCC diagnostic ignored "-Wvariadic-macros" -# pragma GCC diagnostic ignored "-Wunused-variable" -# pragma GCC diagnostic ignored "-Wparentheses" - -# pragma GCC diagnostic push -# pragma GCC diagnostic ignored "-Wpadded" -#endif -#if defined(CATCH_CONFIG_MAIN) || defined(CATCH_CONFIG_RUNNER) -# define CATCH_IMPL -#endif - -#ifdef CATCH_IMPL -# ifndef CLARA_CONFIG_MAIN -# define CLARA_CONFIG_MAIN_NOT_DEFINED -# define CLARA_CONFIG_MAIN -# endif -#endif - -// #included from: internal/catch_notimplemented_exception.h -#define TWOBLUECUBES_CATCH_NOTIMPLEMENTED_EXCEPTION_H_INCLUDED - -// #included from: catch_common.h -#define TWOBLUECUBES_CATCH_COMMON_H_INCLUDED - -// #included from: catch_compiler_capabilities.h -#define TWOBLUECUBES_CATCH_COMPILER_CAPABILITIES_HPP_INCLUDED - -// Detect a number of compiler features - mostly C++11/14 conformance - by compiler -// The following features are defined: -// -// CATCH_CONFIG_CPP11_NULLPTR : is nullptr supported? -// CATCH_CONFIG_CPP11_NOEXCEPT : is noexcept supported? -// CATCH_CONFIG_CPP11_GENERATED_METHODS : The delete and default keywords for compiler generated methods -// CATCH_CONFIG_CPP11_IS_ENUM : std::is_enum is supported? -// CATCH_CONFIG_CPP11_TUPLE : std::tuple is supported -// CATCH_CONFIG_CPP11_LONG_LONG : is long long supported? -// CATCH_CONFIG_CPP11_OVERRIDE : is override supported? -// CATCH_CONFIG_CPP11_UNIQUE_PTR : is unique_ptr supported (otherwise use auto_ptr) -// CATCH_CONFIG_CPP11_SHUFFLE : is std::shuffle supported? -// CATCH_CONFIG_CPP11_TYPE_TRAITS : are type_traits and enable_if supported? - -// CATCH_CONFIG_CPP11_OR_GREATER : Is C++11 supported? - -// CATCH_CONFIG_VARIADIC_MACROS : are variadic macros supported? -// CATCH_CONFIG_COUNTER : is the __COUNTER__ macro supported? -// CATCH_CONFIG_WINDOWS_SEH : is Windows SEH supported? -// CATCH_CONFIG_POSIX_SIGNALS : are POSIX signals supported? -// **************** -// Note to maintainers: if new toggles are added please document them -// in configuration.md, too -// **************** - -// In general each macro has a _NO_ form -// (e.g. CATCH_CONFIG_CPP11_NO_NULLPTR) which disables the feature. -// Many features, at point of detection, define an _INTERNAL_ macro, so they -// can be combined, en-mass, with the _NO_ forms later. - -// All the C++11 features can be disabled with CATCH_CONFIG_NO_CPP11 - -#ifdef __cplusplus - -# if __cplusplus >= 201103L -# define CATCH_CPP11_OR_GREATER -# endif - -# if __cplusplus >= 201402L -# define CATCH_CPP14_OR_GREATER -# endif - -#endif - -#ifdef __clang__ - -# if __has_feature(cxx_nullptr) -# define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR -# endif - -# if __has_feature(cxx_noexcept) -# define CATCH_INTERNAL_CONFIG_CPP11_NOEXCEPT -# endif - -# if defined(CATCH_CPP11_OR_GREATER) -# define CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - _Pragma( "clang diagnostic push" ) \ - _Pragma( "clang diagnostic ignored \"-Wexit-time-destructors\"" ) -# define CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ - _Pragma( "clang diagnostic pop" ) - -# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ - _Pragma( "clang diagnostic push" ) \ - _Pragma( "clang diagnostic ignored \"-Wparentheses\"" ) -# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ - _Pragma( "clang diagnostic pop" ) -# endif - -#endif // __clang__ - -//////////////////////////////////////////////////////////////////////////////// -// We know some environments not to support full POSIX signals -#if defined(__CYGWIN__) || defined(__QNX__) - -# if !defined(CATCH_CONFIG_POSIX_SIGNALS) -# define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS -# endif - -#endif - -#ifdef __OS400__ -# define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS -# define CATCH_CONFIG_COLOUR_NONE -#endif - -//////////////////////////////////////////////////////////////////////////////// -// Cygwin -#ifdef __CYGWIN__ - -// Required for some versions of Cygwin to declare gettimeofday -// see: http://stackoverflow.com/questions/36901803/gettimeofday-not-declared-in-this-scope-cygwin -# define _BSD_SOURCE - -#endif // __CYGWIN__ - -//////////////////////////////////////////////////////////////////////////////// -// Borland -#ifdef __BORLANDC__ - -#endif // __BORLANDC__ - -//////////////////////////////////////////////////////////////////////////////// -// EDG -#ifdef __EDG_VERSION__ - -#endif // __EDG_VERSION__ - -//////////////////////////////////////////////////////////////////////////////// -// Digital Mars -#ifdef __DMC__ - -#endif // __DMC__ - -//////////////////////////////////////////////////////////////////////////////// -// GCC -#ifdef __GNUC__ - -# if __GNUC__ == 4 && __GNUC_MINOR__ >= 6 && defined(__GXX_EXPERIMENTAL_CXX0X__) -# define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR -# endif - -// - otherwise more recent versions define __cplusplus >= 201103L -// and will get picked up below - -#endif // __GNUC__ - -//////////////////////////////////////////////////////////////////////////////// -// Visual C++ -#ifdef _MSC_VER - -#define CATCH_INTERNAL_CONFIG_WINDOWS_SEH - -#if (_MSC_VER >= 1600) -# define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR -# define CATCH_INTERNAL_CONFIG_CPP11_UNIQUE_PTR -#endif - -#if (_MSC_VER >= 1900 ) // (VC++ 13 (VS2015)) -#define CATCH_INTERNAL_CONFIG_CPP11_NOEXCEPT -#define CATCH_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -#define CATCH_INTERNAL_CONFIG_CPP11_SHUFFLE -#define CATCH_INTERNAL_CONFIG_CPP11_TYPE_TRAITS -#endif - -#endif // _MSC_VER - -//////////////////////////////////////////////////////////////////////////////// - -// Use variadic macros if the compiler supports them -#if ( defined _MSC_VER && _MSC_VER > 1400 && !defined __EDGE__) || \ - ( defined __WAVE__ && __WAVE_HAS_VARIADICS ) || \ - ( defined __GNUC__ && __GNUC__ >= 3 ) || \ - ( !defined __cplusplus && __STDC_VERSION__ >= 199901L || __cplusplus >= 201103L ) - -#define CATCH_INTERNAL_CONFIG_VARIADIC_MACROS - -#endif - -// Use __COUNTER__ if the compiler supports it -#if ( defined _MSC_VER && _MSC_VER >= 1300 ) || \ - ( defined __GNUC__ && ( __GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 3 )) ) || \ - ( defined __clang__ && __clang_major__ >= 3 ) - -// Use of __COUNTER__ is suppressed during code analysis in CLion/AppCode 2017.2.x and former, -// because __COUNTER__ is not properly handled by it. -// This does not affect compilation -#if ( !defined __JETBRAINS_IDE__ || __JETBRAINS_IDE__ >= 20170300L ) - #define CATCH_INTERNAL_CONFIG_COUNTER -#endif - -#endif - -//////////////////////////////////////////////////////////////////////////////// -// C++ language feature support - -// catch all support for C++11 -#if defined(CATCH_CPP11_OR_GREATER) - -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_NULLPTR) -# define CATCH_INTERNAL_CONFIG_CPP11_NULLPTR -# endif - -# ifndef CATCH_INTERNAL_CONFIG_CPP11_NOEXCEPT -# define CATCH_INTERNAL_CONFIG_CPP11_NOEXCEPT -# endif - -# ifndef CATCH_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -# define CATCH_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -# endif - -# ifndef CATCH_INTERNAL_CONFIG_CPP11_IS_ENUM -# define CATCH_INTERNAL_CONFIG_CPP11_IS_ENUM -# endif - -# ifndef CATCH_INTERNAL_CONFIG_CPP11_TUPLE -# define CATCH_INTERNAL_CONFIG_CPP11_TUPLE -# endif - -# ifndef CATCH_INTERNAL_CONFIG_VARIADIC_MACROS -# define CATCH_INTERNAL_CONFIG_VARIADIC_MACROS -# endif - -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_LONG_LONG) -# define CATCH_INTERNAL_CONFIG_CPP11_LONG_LONG -# endif - -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_OVERRIDE) -# define CATCH_INTERNAL_CONFIG_CPP11_OVERRIDE -# endif -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_UNIQUE_PTR) -# define CATCH_INTERNAL_CONFIG_CPP11_UNIQUE_PTR -# endif -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_SHUFFLE) -# define CATCH_INTERNAL_CONFIG_CPP11_SHUFFLE -# endif -# if !defined(CATCH_INTERNAL_CONFIG_CPP11_TYPE_TRAITS) -# define CATCH_INTERNAL_CONFIG_CPP11_TYPE_TRAITS -# endif - -#endif // __cplusplus >= 201103L - -// Now set the actual defines based on the above + anything the user has configured -#if defined(CATCH_INTERNAL_CONFIG_CPP11_NULLPTR) && !defined(CATCH_CONFIG_CPP11_NO_NULLPTR) && !defined(CATCH_CONFIG_CPP11_NULLPTR) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_NULLPTR -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_NOEXCEPT) && !defined(CATCH_CONFIG_CPP11_NO_NOEXCEPT) && !defined(CATCH_CONFIG_CPP11_NOEXCEPT) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_NOEXCEPT -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_GENERATED_METHODS) && !defined(CATCH_CONFIG_CPP11_NO_GENERATED_METHODS) && !defined(CATCH_CONFIG_CPP11_GENERATED_METHODS) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_GENERATED_METHODS -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_IS_ENUM) && !defined(CATCH_CONFIG_CPP11_NO_IS_ENUM) && !defined(CATCH_CONFIG_CPP11_IS_ENUM) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_IS_ENUM -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_TUPLE) && !defined(CATCH_CONFIG_CPP11_NO_TUPLE) && !defined(CATCH_CONFIG_CPP11_TUPLE) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_TUPLE -#endif -#if defined(CATCH_INTERNAL_CONFIG_VARIADIC_MACROS) && !defined(CATCH_CONFIG_NO_VARIADIC_MACROS) && !defined(CATCH_CONFIG_VARIADIC_MACROS) -# define CATCH_CONFIG_VARIADIC_MACROS -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_LONG_LONG) && !defined(CATCH_CONFIG_CPP11_NO_LONG_LONG) && !defined(CATCH_CONFIG_CPP11_LONG_LONG) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_LONG_LONG -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_OVERRIDE) && !defined(CATCH_CONFIG_CPP11_NO_OVERRIDE) && !defined(CATCH_CONFIG_CPP11_OVERRIDE) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_OVERRIDE -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_UNIQUE_PTR) && !defined(CATCH_CONFIG_CPP11_NO_UNIQUE_PTR) && !defined(CATCH_CONFIG_CPP11_UNIQUE_PTR) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_UNIQUE_PTR -#endif -#if defined(CATCH_INTERNAL_CONFIG_COUNTER) && !defined(CATCH_CONFIG_NO_COUNTER) && !defined(CATCH_CONFIG_COUNTER) -# define CATCH_CONFIG_COUNTER -#endif -#if defined(CATCH_INTERNAL_CONFIG_CPP11_SHUFFLE) && !defined(CATCH_CONFIG_CPP11_NO_SHUFFLE) && !defined(CATCH_CONFIG_CPP11_SHUFFLE) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_SHUFFLE -#endif -# if defined(CATCH_INTERNAL_CONFIG_CPP11_TYPE_TRAITS) && !defined(CATCH_CONFIG_CPP11_NO_TYPE_TRAITS) && !defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) && !defined(CATCH_CONFIG_NO_CPP11) -# define CATCH_CONFIG_CPP11_TYPE_TRAITS -# endif -#if defined(CATCH_INTERNAL_CONFIG_WINDOWS_SEH) && !defined(CATCH_CONFIG_NO_WINDOWS_SEH) && !defined(CATCH_CONFIG_WINDOWS_SEH) -# define CATCH_CONFIG_WINDOWS_SEH -#endif -// This is set by default, because we assume that unix compilers are posix-signal-compatible by default. -#if !defined(CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_POSIX_SIGNALS) -# define CATCH_CONFIG_POSIX_SIGNALS -#endif - -#if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) -# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS -# define CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS -#endif -#if !defined(CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS) -# define CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS -# define CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS -#endif - -// noexcept support: -#if defined(CATCH_CONFIG_CPP11_NOEXCEPT) && !defined(CATCH_NOEXCEPT) -# define CATCH_NOEXCEPT noexcept -# define CATCH_NOEXCEPT_IS(x) noexcept(x) -#else -# define CATCH_NOEXCEPT throw() -# define CATCH_NOEXCEPT_IS(x) -#endif - -// nullptr support -#ifdef CATCH_CONFIG_CPP11_NULLPTR -# define CATCH_NULL nullptr -#else -# define CATCH_NULL NULL -#endif - -// override support -#ifdef CATCH_CONFIG_CPP11_OVERRIDE -# define CATCH_OVERRIDE override -#else -# define CATCH_OVERRIDE -#endif - -// unique_ptr support -#ifdef CATCH_CONFIG_CPP11_UNIQUE_PTR -# define CATCH_AUTO_PTR( T ) std::unique_ptr -#else -# define CATCH_AUTO_PTR( T ) std::auto_ptr -#endif - -#define INTERNAL_CATCH_UNIQUE_NAME_LINE2( name, line ) name##line -#define INTERNAL_CATCH_UNIQUE_NAME_LINE( name, line ) INTERNAL_CATCH_UNIQUE_NAME_LINE2( name, line ) -#ifdef CATCH_CONFIG_COUNTER -# define INTERNAL_CATCH_UNIQUE_NAME( name ) INTERNAL_CATCH_UNIQUE_NAME_LINE( name, __COUNTER__ ) -#else -# define INTERNAL_CATCH_UNIQUE_NAME( name ) INTERNAL_CATCH_UNIQUE_NAME_LINE( name, __LINE__ ) -#endif - -#define INTERNAL_CATCH_STRINGIFY2( expr ) #expr -#define INTERNAL_CATCH_STRINGIFY( expr ) INTERNAL_CATCH_STRINGIFY2( expr ) - -#include -#include - -namespace Catch { - - struct IConfig; - - struct CaseSensitive { enum Choice { - Yes, - No - }; }; - - class NonCopyable { -#ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - NonCopyable( NonCopyable const& ) = delete; - NonCopyable( NonCopyable && ) = delete; - NonCopyable& operator = ( NonCopyable const& ) = delete; - NonCopyable& operator = ( NonCopyable && ) = delete; -#else - NonCopyable( NonCopyable const& info ); - NonCopyable& operator = ( NonCopyable const& ); -#endif - - protected: - NonCopyable() {} - virtual ~NonCopyable(); - }; - - class SafeBool { - public: - typedef void (SafeBool::*type)() const; - - static type makeSafe( bool value ) { - return value ? &SafeBool::trueValue : 0; - } - private: - void trueValue() const {} - }; - - template - void deleteAll( ContainerT& container ) { - typename ContainerT::const_iterator it = container.begin(); - typename ContainerT::const_iterator itEnd = container.end(); - for(; it != itEnd; ++it ) - delete *it; - } - template - void deleteAllValues( AssociativeContainerT& container ) { - typename AssociativeContainerT::const_iterator it = container.begin(); - typename AssociativeContainerT::const_iterator itEnd = container.end(); - for(; it != itEnd; ++it ) - delete it->second; - } - - bool startsWith( std::string const& s, std::string const& prefix ); - bool startsWith( std::string const& s, char prefix ); - bool endsWith( std::string const& s, std::string const& suffix ); - bool endsWith( std::string const& s, char suffix ); - bool contains( std::string const& s, std::string const& infix ); - void toLowerInPlace( std::string& s ); - std::string toLower( std::string const& s ); - std::string trim( std::string const& str ); - bool replaceInPlace( std::string& str, std::string const& replaceThis, std::string const& withThis ); - - struct pluralise { - pluralise( std::size_t count, std::string const& label ); - - friend std::ostream& operator << ( std::ostream& os, pluralise const& pluraliser ); - - std::size_t m_count; - std::string m_label; - }; - - struct SourceLineInfo { - - SourceLineInfo(); - SourceLineInfo( char const* _file, std::size_t _line ); -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - SourceLineInfo(SourceLineInfo const& other) = default; - SourceLineInfo( SourceLineInfo && ) = default; - SourceLineInfo& operator = ( SourceLineInfo const& ) = default; - SourceLineInfo& operator = ( SourceLineInfo && ) = default; -# endif - bool empty() const; - bool operator == ( SourceLineInfo const& other ) const; - bool operator < ( SourceLineInfo const& other ) const; - - char const* file; - std::size_t line; - }; - - std::ostream& operator << ( std::ostream& os, SourceLineInfo const& info ); - - // This is just here to avoid compiler warnings with macro constants and boolean literals - inline bool isTrue( bool value ){ return value; } - inline bool alwaysTrue() { return true; } - inline bool alwaysFalse() { return false; } - - void throwLogicError( std::string const& message, SourceLineInfo const& locationInfo ); - - void seedRng( IConfig const& config ); - unsigned int rngSeed(); - - // Use this in variadic streaming macros to allow - // >> +StreamEndStop - // as well as - // >> stuff +StreamEndStop - struct StreamEndStop { - std::string operator+() { - return std::string(); - } - }; - template - T const& operator + ( T const& value, StreamEndStop ) { - return value; - } -} - -#define CATCH_INTERNAL_LINEINFO ::Catch::SourceLineInfo( __FILE__, static_cast( __LINE__ ) ) -#define CATCH_INTERNAL_ERROR( msg ) ::Catch::throwLogicError( msg, CATCH_INTERNAL_LINEINFO ); - -namespace Catch { - - class NotImplementedException : public std::exception - { - public: - NotImplementedException( SourceLineInfo const& lineInfo ); - - virtual ~NotImplementedException() CATCH_NOEXCEPT {} - - virtual const char* what() const CATCH_NOEXCEPT; - - private: - std::string m_what; - SourceLineInfo m_lineInfo; - }; - -} // end namespace Catch - -/////////////////////////////////////////////////////////////////////////////// -#define CATCH_NOT_IMPLEMENTED throw Catch::NotImplementedException( CATCH_INTERNAL_LINEINFO ) - -// #included from: internal/catch_context.h -#define TWOBLUECUBES_CATCH_CONTEXT_H_INCLUDED - -// #included from: catch_interfaces_generators.h -#define TWOBLUECUBES_CATCH_INTERFACES_GENERATORS_H_INCLUDED - -#include - -namespace Catch { - - struct IGeneratorInfo { - virtual ~IGeneratorInfo(); - virtual bool moveNext() = 0; - virtual std::size_t getCurrentIndex() const = 0; - }; - - struct IGeneratorsForTest { - virtual ~IGeneratorsForTest(); - - virtual IGeneratorInfo& getGeneratorInfo( std::string const& fileInfo, std::size_t size ) = 0; - virtual bool moveNext() = 0; - }; - - IGeneratorsForTest* createGeneratorsForTest(); - -} // end namespace Catch - -// #included from: catch_ptr.hpp -#define TWOBLUECUBES_CATCH_PTR_HPP_INCLUDED - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wpadded" -#endif - -namespace Catch { - - // An intrusive reference counting smart pointer. - // T must implement addRef() and release() methods - // typically implementing the IShared interface - template - class Ptr { - public: - Ptr() : m_p( CATCH_NULL ){} - Ptr( T* p ) : m_p( p ){ - if( m_p ) - m_p->addRef(); - } - Ptr( Ptr const& other ) : m_p( other.m_p ){ - if( m_p ) - m_p->addRef(); - } - ~Ptr(){ - if( m_p ) - m_p->release(); - } - void reset() { - if( m_p ) - m_p->release(); - m_p = CATCH_NULL; - } - Ptr& operator = ( T* p ){ - Ptr temp( p ); - swap( temp ); - return *this; - } - Ptr& operator = ( Ptr const& other ){ - Ptr temp( other ); - swap( temp ); - return *this; - } - void swap( Ptr& other ) { std::swap( m_p, other.m_p ); } - T* get() const{ return m_p; } - T& operator*() const { return *m_p; } - T* operator->() const { return m_p; } - bool operator !() const { return m_p == CATCH_NULL; } - operator SafeBool::type() const { return SafeBool::makeSafe( m_p != CATCH_NULL ); } - - private: - T* m_p; - }; - - struct IShared : NonCopyable { - virtual ~IShared(); - virtual void addRef() const = 0; - virtual void release() const = 0; - }; - - template - struct SharedImpl : T { - - SharedImpl() : m_rc( 0 ){} - - virtual void addRef() const { - ++m_rc; - } - virtual void release() const { - if( --m_rc == 0 ) - delete this; - } - - mutable unsigned int m_rc; - }; - -} // end namespace Catch - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - -namespace Catch { - - class TestCase; - class Stream; - struct IResultCapture; - struct IRunner; - struct IGeneratorsForTest; - struct IConfig; - - struct IContext - { - virtual ~IContext(); - - virtual IResultCapture* getResultCapture() = 0; - virtual IRunner* getRunner() = 0; - virtual size_t getGeneratorIndex( std::string const& fileInfo, size_t totalSize ) = 0; - virtual bool advanceGeneratorsForCurrentTest() = 0; - virtual Ptr getConfig() const = 0; - }; - - struct IMutableContext : IContext - { - virtual ~IMutableContext(); - virtual void setResultCapture( IResultCapture* resultCapture ) = 0; - virtual void setRunner( IRunner* runner ) = 0; - virtual void setConfig( Ptr const& config ) = 0; - }; - - IContext& getCurrentContext(); - IMutableContext& getCurrentMutableContext(); - void cleanUpContext(); - Stream createStream( std::string const& streamName ); - -} - -// #included from: internal/catch_test_registry.hpp -#define TWOBLUECUBES_CATCH_TEST_REGISTRY_HPP_INCLUDED - -// #included from: catch_interfaces_testcase.h -#define TWOBLUECUBES_CATCH_INTERFACES_TESTCASE_H_INCLUDED - -#include - -namespace Catch { - - class TestSpec; - - struct ITestCase : IShared { - virtual void invoke () const = 0; - protected: - virtual ~ITestCase(); - }; - - class TestCase; - struct IConfig; - - struct ITestCaseRegistry { - virtual ~ITestCaseRegistry(); - virtual std::vector const& getAllTests() const = 0; - virtual std::vector const& getAllTestsSorted( IConfig const& config ) const = 0; - }; - - bool matchTest( TestCase const& testCase, TestSpec const& testSpec, IConfig const& config ); - std::vector filterTests( std::vector const& testCases, TestSpec const& testSpec, IConfig const& config ); - std::vector const& getAllTestCasesSorted( IConfig const& config ); - -} - -namespace Catch { - -template -class MethodTestCase : public SharedImpl { - -public: - MethodTestCase( void (C::*method)() ) : m_method( method ) {} - - virtual void invoke() const { - C obj; - (obj.*m_method)(); - } - -private: - virtual ~MethodTestCase() {} - - void (C::*m_method)(); -}; - -typedef void(*TestFunction)(); - -struct NameAndDesc { - NameAndDesc( const char* _name = "", const char* _description= "" ) - : name( _name ), description( _description ) - {} - - const char* name; - const char* description; -}; - -void registerTestCase - ( ITestCase* testCase, - char const* className, - NameAndDesc const& nameAndDesc, - SourceLineInfo const& lineInfo ); - -struct AutoReg { - - AutoReg - ( TestFunction function, - SourceLineInfo const& lineInfo, - NameAndDesc const& nameAndDesc ); - - template - AutoReg - ( void (C::*method)(), - char const* className, - NameAndDesc const& nameAndDesc, - SourceLineInfo const& lineInfo ) { - - registerTestCase - ( new MethodTestCase( method ), - className, - nameAndDesc, - lineInfo ); - } - - ~AutoReg(); - -private: - AutoReg( AutoReg const& ); - void operator= ( AutoReg const& ); -}; - -void registerTestCaseFunction - ( TestFunction function, - SourceLineInfo const& lineInfo, - NameAndDesc const& nameAndDesc ); - -} // end namespace Catch - -#ifdef CATCH_CONFIG_VARIADIC_MACROS - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_TESTCASE2( TestName, ... ) \ - static void TestName(); \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &TestName, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); } /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ - static void TestName() - #define INTERNAL_CATCH_TESTCASE( ... ) \ - INTERNAL_CATCH_TESTCASE2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), __VA_ARGS__ ) - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, ... ) \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( __VA_ARGS__ ), CATCH_INTERNAL_LINEINFO ); } /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_TEST_CASE_METHOD2( TestName, ClassName, ... )\ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ \ - struct TestName : ClassName{ \ - void test(); \ - }; \ - Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar ) ( &TestName::test, #ClassName, Catch::NameAndDesc( __VA_ARGS__ ), CATCH_INTERNAL_LINEINFO ); /* NOLINT */ \ - } \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ - void TestName::test() - #define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, ... ) \ - INTERNAL_CATCH_TEST_CASE_METHOD2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), ClassName, __VA_ARGS__ ) - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_REGISTER_TESTCASE( Function, ... ) \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( __VA_ARGS__ ) ); /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS - -#else - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_TESTCASE2( TestName, Name, Desc ) \ - static void TestName(); \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &TestName, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( Name, Desc ) ); } /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ - static void TestName() - #define INTERNAL_CATCH_TESTCASE( Name, Desc ) \ - INTERNAL_CATCH_TESTCASE2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), Name, Desc ) - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_METHOD_AS_TEST_CASE( QualifiedMethod, Name, Desc ) \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar )( &QualifiedMethod, "&" #QualifiedMethod, Catch::NameAndDesc( Name, Desc ), CATCH_INTERNAL_LINEINFO ); } /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_TEST_CASE_METHOD2( TestCaseName, ClassName, TestName, Desc )\ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - namespace{ \ - struct TestCaseName : ClassName{ \ - void test(); \ - }; \ - Catch::AutoReg INTERNAL_CATCH_UNIQUE_NAME( autoRegistrar ) ( &TestCaseName::test, #ClassName, Catch::NameAndDesc( TestName, Desc ), CATCH_INTERNAL_LINEINFO ); /* NOLINT */ \ - } \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS \ - void TestCaseName::test() - #define INTERNAL_CATCH_TEST_CASE_METHOD( ClassName, TestName, Desc )\ - INTERNAL_CATCH_TEST_CASE_METHOD2( INTERNAL_CATCH_UNIQUE_NAME( ____C_A_T_C_H____T_E_S_T____ ), ClassName, TestName, Desc ) - - /////////////////////////////////////////////////////////////////////////////// - #define INTERNAL_CATCH_REGISTER_TESTCASE( Function, Name, Desc ) \ - CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS \ - Catch::AutoReg( Function, CATCH_INTERNAL_LINEINFO, Catch::NameAndDesc( Name, Desc ) ); /* NOLINT */ \ - CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS - -#endif - -// #included from: internal/catch_capture.hpp -#define TWOBLUECUBES_CATCH_CAPTURE_HPP_INCLUDED - -// #included from: catch_result_builder.h -#define TWOBLUECUBES_CATCH_RESULT_BUILDER_H_INCLUDED - -// #included from: catch_result_type.h -#define TWOBLUECUBES_CATCH_RESULT_TYPE_H_INCLUDED - -namespace Catch { - - // ResultWas::OfType enum - struct ResultWas { enum OfType { - Unknown = -1, - Ok = 0, - Info = 1, - Warning = 2, - - FailureBit = 0x10, - - ExpressionFailed = FailureBit | 1, - ExplicitFailure = FailureBit | 2, - - Exception = 0x100 | FailureBit, - - ThrewException = Exception | 1, - DidntThrowException = Exception | 2, - - FatalErrorCondition = 0x200 | FailureBit - - }; }; - - inline bool isOk( ResultWas::OfType resultType ) { - return ( resultType & ResultWas::FailureBit ) == 0; - } - inline bool isJustInfo( int flags ) { - return flags == ResultWas::Info; - } - - // ResultDisposition::Flags enum - struct ResultDisposition { enum Flags { - Normal = 0x01, - - ContinueOnFailure = 0x02, // Failures fail test, but execution continues - FalseTest = 0x04, // Prefix expression with ! - SuppressFail = 0x08 // Failures are reported but do not fail the test - }; }; - - inline ResultDisposition::Flags operator | ( ResultDisposition::Flags lhs, ResultDisposition::Flags rhs ) { - return static_cast( static_cast( lhs ) | static_cast( rhs ) ); - } - - inline bool shouldContinueOnFailure( int flags ) { return ( flags & ResultDisposition::ContinueOnFailure ) != 0; } - inline bool isFalseTest( int flags ) { return ( flags & ResultDisposition::FalseTest ) != 0; } - inline bool shouldSuppressFailure( int flags ) { return ( flags & ResultDisposition::SuppressFail ) != 0; } - -} // end namespace Catch - -// #included from: catch_assertionresult.h -#define TWOBLUECUBES_CATCH_ASSERTIONRESULT_H_INCLUDED - -#include - -namespace Catch { - - struct STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison; - - struct DecomposedExpression - { - virtual ~DecomposedExpression() {} - virtual bool isBinaryExpression() const { - return false; - } - virtual void reconstructExpression( std::string& dest ) const = 0; - - // Only simple binary comparisons can be decomposed. - // If more complex check is required then wrap sub-expressions in parentheses. - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator + ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator - ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator * ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator / ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator % ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator && ( T const& ); - template STATIC_ASSERT_Expression_Too_Complex_Please_Rewrite_As_Binary_Comparison& operator || ( T const& ); - - private: - DecomposedExpression& operator = (DecomposedExpression const&); - }; - - struct AssertionInfo - { - AssertionInfo(); - AssertionInfo( char const * _macroName, - SourceLineInfo const& _lineInfo, - char const * _capturedExpression, - ResultDisposition::Flags _resultDisposition, - char const * _secondArg = ""); - - char const * macroName; - SourceLineInfo lineInfo; - char const * capturedExpression; - ResultDisposition::Flags resultDisposition; - char const * secondArg; - }; - - struct AssertionResultData - { - AssertionResultData() : decomposedExpression( CATCH_NULL ) - , resultType( ResultWas::Unknown ) - , negated( false ) - , parenthesized( false ) {} - - void negate( bool parenthesize ) { - negated = !negated; - parenthesized = parenthesize; - if( resultType == ResultWas::Ok ) - resultType = ResultWas::ExpressionFailed; - else if( resultType == ResultWas::ExpressionFailed ) - resultType = ResultWas::Ok; - } - - std::string const& reconstructExpression() const { - if( decomposedExpression != CATCH_NULL ) { - decomposedExpression->reconstructExpression( reconstructedExpression ); - if( parenthesized ) { - reconstructedExpression.insert( 0, 1, '(' ); - reconstructedExpression.append( 1, ')' ); - } - if( negated ) { - reconstructedExpression.insert( 0, 1, '!' ); - } - decomposedExpression = CATCH_NULL; - } - return reconstructedExpression; - } - - mutable DecomposedExpression const* decomposedExpression; - mutable std::string reconstructedExpression; - std::string message; - ResultWas::OfType resultType; - bool negated; - bool parenthesized; - }; - - class AssertionResult { - public: - AssertionResult(); - AssertionResult( AssertionInfo const& info, AssertionResultData const& data ); - ~AssertionResult(); -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - AssertionResult( AssertionResult const& ) = default; - AssertionResult( AssertionResult && ) = default; - AssertionResult& operator = ( AssertionResult const& ) = default; - AssertionResult& operator = ( AssertionResult && ) = default; -# endif - - bool isOk() const; - bool succeeded() const; - ResultWas::OfType getResultType() const; - bool hasExpression() const; - bool hasMessage() const; - std::string getExpression() const; - std::string getExpressionInMacro() const; - bool hasExpandedExpression() const; - std::string getExpandedExpression() const; - std::string getMessage() const; - SourceLineInfo getSourceInfo() const; - std::string getTestMacroName() const; - void discardDecomposedExpression() const; - void expandDecomposedExpression() const; - - protected: - AssertionInfo m_info; - AssertionResultData m_resultData; - }; - -} // end namespace Catch - -// #included from: catch_matchers.hpp -#define TWOBLUECUBES_CATCH_MATCHERS_HPP_INCLUDED - -namespace Catch { -namespace Matchers { - namespace Impl { - - template struct MatchAllOf; - template struct MatchAnyOf; - template struct MatchNotOf; - - class MatcherUntypedBase { - public: - std::string toString() const { - if( m_cachedToString.empty() ) - m_cachedToString = describe(); - return m_cachedToString; - } - - protected: - virtual ~MatcherUntypedBase(); - virtual std::string describe() const = 0; - mutable std::string m_cachedToString; - private: - MatcherUntypedBase& operator = ( MatcherUntypedBase const& ); - }; - - template - struct MatcherMethod { - virtual bool match( ObjectT const& arg ) const = 0; - }; - template - struct MatcherMethod { - virtual bool match( PtrT* arg ) const = 0; - }; - - template - struct MatcherBase : MatcherUntypedBase, MatcherMethod { - - MatchAllOf operator && ( MatcherBase const& other ) const; - MatchAnyOf operator || ( MatcherBase const& other ) const; - MatchNotOf operator ! () const; - }; - - template - struct MatchAllOf : MatcherBase { - virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { - for( std::size_t i = 0; i < m_matchers.size(); ++i ) { - if (!m_matchers[i]->match(arg)) - return false; - } - return true; - } - virtual std::string describe() const CATCH_OVERRIDE { - std::string description; - description.reserve( 4 + m_matchers.size()*32 ); - description += "( "; - for( std::size_t i = 0; i < m_matchers.size(); ++i ) { - if( i != 0 ) - description += " and "; - description += m_matchers[i]->toString(); - } - description += " )"; - return description; - } - - MatchAllOf& operator && ( MatcherBase const& other ) { - m_matchers.push_back( &other ); - return *this; - } - - std::vector const*> m_matchers; - }; - template - struct MatchAnyOf : MatcherBase { - - virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { - for( std::size_t i = 0; i < m_matchers.size(); ++i ) { - if (m_matchers[i]->match(arg)) - return true; - } - return false; - } - virtual std::string describe() const CATCH_OVERRIDE { - std::string description; - description.reserve( 4 + m_matchers.size()*32 ); - description += "( "; - for( std::size_t i = 0; i < m_matchers.size(); ++i ) { - if( i != 0 ) - description += " or "; - description += m_matchers[i]->toString(); - } - description += " )"; - return description; - } - - MatchAnyOf& operator || ( MatcherBase const& other ) { - m_matchers.push_back( &other ); - return *this; - } - - std::vector const*> m_matchers; - }; - - template - struct MatchNotOf : MatcherBase { - - MatchNotOf( MatcherBase const& underlyingMatcher ) : m_underlyingMatcher( underlyingMatcher ) {} - - virtual bool match( ArgT const& arg ) const CATCH_OVERRIDE { - return !m_underlyingMatcher.match( arg ); - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "not " + m_underlyingMatcher.toString(); - } - MatcherBase const& m_underlyingMatcher; - }; - - template - MatchAllOf MatcherBase::operator && ( MatcherBase const& other ) const { - return MatchAllOf() && *this && other; - } - template - MatchAnyOf MatcherBase::operator || ( MatcherBase const& other ) const { - return MatchAnyOf() || *this || other; - } - template - MatchNotOf MatcherBase::operator ! () const { - return MatchNotOf( *this ); - } - - } // namespace Impl - - // The following functions create the actual matcher objects. - // This allows the types to be inferred - // - deprecated: prefer ||, && and ! - template - Impl::MatchNotOf Not( Impl::MatcherBase const& underlyingMatcher ) { - return Impl::MatchNotOf( underlyingMatcher ); - } - template - Impl::MatchAllOf AllOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2 ) { - return Impl::MatchAllOf() && m1 && m2; - } - template - Impl::MatchAllOf AllOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2, Impl::MatcherBase const& m3 ) { - return Impl::MatchAllOf() && m1 && m2 && m3; - } - template - Impl::MatchAnyOf AnyOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2 ) { - return Impl::MatchAnyOf() || m1 || m2; - } - template - Impl::MatchAnyOf AnyOf( Impl::MatcherBase const& m1, Impl::MatcherBase const& m2, Impl::MatcherBase const& m3 ) { - return Impl::MatchAnyOf() || m1 || m2 || m3; - } - -} // namespace Matchers - -using namespace Matchers; -using Matchers::Impl::MatcherBase; - -} // namespace Catch - -namespace Catch { - - struct TestFailureException{}; - - template class ExpressionLhs; - - struct CopyableStream { - CopyableStream() {} - CopyableStream( CopyableStream const& other ) { - oss << other.oss.str(); - } - CopyableStream& operator=( CopyableStream const& other ) { - oss.str(std::string()); - oss << other.oss.str(); - return *this; - } - std::ostringstream oss; - }; - - class ResultBuilder : public DecomposedExpression { - public: - ResultBuilder( char const* macroName, - SourceLineInfo const& lineInfo, - char const* capturedExpression, - ResultDisposition::Flags resultDisposition, - char const* secondArg = "" ); - ~ResultBuilder(); - - template - ExpressionLhs operator <= ( T const& operand ); - ExpressionLhs operator <= ( bool value ); - - template - ResultBuilder& operator << ( T const& value ) { - stream().oss << value; - return *this; - } - - ResultBuilder& setResultType( ResultWas::OfType result ); - ResultBuilder& setResultType( bool result ); - - void endExpression( DecomposedExpression const& expr ); - - virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE; - - AssertionResult build() const; - AssertionResult build( DecomposedExpression const& expr ) const; - - void useActiveException( ResultDisposition::Flags resultDisposition = ResultDisposition::Normal ); - void captureResult( ResultWas::OfType resultType ); - void captureExpression(); - void captureExpectedException( std::string const& expectedMessage ); - void captureExpectedException( Matchers::Impl::MatcherBase const& matcher ); - void handleResult( AssertionResult const& result ); - void react(); - bool shouldDebugBreak() const; - bool allowThrows() const; - - template - void captureMatch( ArgT const& arg, MatcherT const& matcher, char const* matcherString ); - - void setExceptionGuard(); - void unsetExceptionGuard(); - - private: - AssertionInfo m_assertionInfo; - AssertionResultData m_data; - - CopyableStream &stream() - { - if(!m_usedStream) - { - m_usedStream = true; - m_stream().oss.str(""); - } - return m_stream(); - } - - static CopyableStream &m_stream() - { - static CopyableStream s; - return s; - } - - bool m_shouldDebugBreak; - bool m_shouldThrow; - bool m_guardException; - bool m_usedStream; - }; - -} // namespace Catch - -// Include after due to circular dependency: -// #included from: catch_expression_lhs.hpp -#define TWOBLUECUBES_CATCH_EXPRESSION_LHS_HPP_INCLUDED - -// #included from: catch_evaluate.hpp -#define TWOBLUECUBES_CATCH_EVALUATE_HPP_INCLUDED - -#ifdef _MSC_VER -#pragma warning(push) -#pragma warning(disable:4389) // '==' : signed/unsigned mismatch -#pragma warning(disable:4018) // more "signed/unsigned mismatch" -#pragma warning(disable:4312) // Converting int to T* using reinterpret_cast (issue on x64 platform) -#endif - -#include - -namespace Catch { -namespace Internal { - - enum Operator { - IsEqualTo, - IsNotEqualTo, - IsLessThan, - IsGreaterThan, - IsLessThanOrEqualTo, - IsGreaterThanOrEqualTo - }; - - template struct OperatorTraits { static const char* getName(){ return "*error*"; } }; - template<> struct OperatorTraits { static const char* getName(){ return "=="; } }; - template<> struct OperatorTraits { static const char* getName(){ return "!="; } }; - template<> struct OperatorTraits { static const char* getName(){ return "<"; } }; - template<> struct OperatorTraits { static const char* getName(){ return ">"; } }; - template<> struct OperatorTraits { static const char* getName(){ return "<="; } }; - template<> struct OperatorTraits{ static const char* getName(){ return ">="; } }; - - template - T& opCast(T const& t) { return const_cast(t); } - -// nullptr_t support based on pull request #154 from Konstantin Baumann -#ifdef CATCH_CONFIG_CPP11_NULLPTR - inline std::nullptr_t opCast(std::nullptr_t) { return nullptr; } -#endif // CATCH_CONFIG_CPP11_NULLPTR - - // So the compare overloads can be operator agnostic we convey the operator as a template - // enum, which is used to specialise an Evaluator for doing the comparison. - template - struct Evaluator{}; - - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs) { - return bool( opCast( lhs ) == opCast( rhs ) ); - } - }; - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs ) { - return bool( opCast( lhs ) != opCast( rhs ) ); - } - }; - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs ) { - return bool( opCast( lhs ) < opCast( rhs ) ); - } - }; - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs ) { - return bool( opCast( lhs ) > opCast( rhs ) ); - } - }; - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs ) { - return bool( opCast( lhs ) >= opCast( rhs ) ); - } - }; - template - struct Evaluator { - static bool evaluate( T1 const& lhs, T2 const& rhs ) { - return bool( opCast( lhs ) <= opCast( rhs ) ); - } - }; - - template - bool applyEvaluator( T1 const& lhs, T2 const& rhs ) { - return Evaluator::evaluate( lhs, rhs ); - } - - // This level of indirection allows us to specialise for integer types - // to avoid signed/ unsigned warnings - - // "base" overload - template - bool compare( T1 const& lhs, T2 const& rhs ) { - return Evaluator::evaluate( lhs, rhs ); - } - - // unsigned X to int - template bool compare( unsigned int lhs, int rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - template bool compare( unsigned long lhs, int rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - template bool compare( unsigned char lhs, int rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - - // unsigned X to long - template bool compare( unsigned int lhs, long rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - template bool compare( unsigned long lhs, long rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - template bool compare( unsigned char lhs, long rhs ) { - return applyEvaluator( lhs, static_cast( rhs ) ); - } - - // int to unsigned X - template bool compare( int lhs, unsigned int rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( int lhs, unsigned long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( int lhs, unsigned char rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - - // long to unsigned X - template bool compare( long lhs, unsigned int rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( long lhs, unsigned long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( long lhs, unsigned char rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - - // pointer to long (when comparing against NULL) - template bool compare( long lhs, T* rhs ) { - return Evaluator::evaluate( reinterpret_cast( lhs ), rhs ); - } - template bool compare( T* lhs, long rhs ) { - return Evaluator::evaluate( lhs, reinterpret_cast( rhs ) ); - } - - // pointer to int (when comparing against NULL) - template bool compare( int lhs, T* rhs ) { - return Evaluator::evaluate( reinterpret_cast( lhs ), rhs ); - } - template bool compare( T* lhs, int rhs ) { - return Evaluator::evaluate( lhs, reinterpret_cast( rhs ) ); - } - -#ifdef CATCH_CONFIG_CPP11_LONG_LONG - // long long to unsigned X - template bool compare( long long lhs, unsigned int rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( long long lhs, unsigned long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( long long lhs, unsigned long long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( long long lhs, unsigned char rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - - // unsigned long long to X - template bool compare( unsigned long long lhs, int rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( unsigned long long lhs, long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( unsigned long long lhs, long long rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - template bool compare( unsigned long long lhs, char rhs ) { - return applyEvaluator( static_cast( lhs ), rhs ); - } - - // pointer to long long (when comparing against NULL) - template bool compare( long long lhs, T* rhs ) { - return Evaluator::evaluate( reinterpret_cast( lhs ), rhs ); - } - template bool compare( T* lhs, long long rhs ) { - return Evaluator::evaluate( lhs, reinterpret_cast( rhs ) ); - } -#endif // CATCH_CONFIG_CPP11_LONG_LONG - -#ifdef CATCH_CONFIG_CPP11_NULLPTR - // pointer to nullptr_t (when comparing against nullptr) - template bool compare( std::nullptr_t, T* rhs ) { - return Evaluator::evaluate( nullptr, rhs ); - } - template bool compare( T* lhs, std::nullptr_t ) { - return Evaluator::evaluate( lhs, nullptr ); - } -#endif // CATCH_CONFIG_CPP11_NULLPTR - -} // end of namespace Internal -} // end of namespace Catch - -#ifdef _MSC_VER -#pragma warning(pop) -#endif - -// #included from: catch_tostring.h -#define TWOBLUECUBES_CATCH_TOSTRING_H_INCLUDED - -#include -#include -#include -#include -#include - -#ifdef __OBJC__ -// #included from: catch_objc_arc.hpp -#define TWOBLUECUBES_CATCH_OBJC_ARC_HPP_INCLUDED - -#import - -#ifdef __has_feature -#define CATCH_ARC_ENABLED __has_feature(objc_arc) -#else -#define CATCH_ARC_ENABLED 0 -#endif - -void arcSafeRelease( NSObject* obj ); -id performOptionalSelector( id obj, SEL sel ); - -#if !CATCH_ARC_ENABLED -inline void arcSafeRelease( NSObject* obj ) { - [obj release]; -} -inline id performOptionalSelector( id obj, SEL sel ) { - if( [obj respondsToSelector: sel] ) - return [obj performSelector: sel]; - return nil; -} -#define CATCH_UNSAFE_UNRETAINED -#define CATCH_ARC_STRONG -#else -inline void arcSafeRelease( NSObject* ){} -inline id performOptionalSelector( id obj, SEL sel ) { -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Warc-performSelector-leaks" -#endif - if( [obj respondsToSelector: sel] ) - return [obj performSelector: sel]; -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - return nil; -} -#define CATCH_UNSAFE_UNRETAINED __unsafe_unretained -#define CATCH_ARC_STRONG __strong -#endif - -#endif - -#ifdef CATCH_CONFIG_CPP11_TUPLE -#include -#endif - -#ifdef CATCH_CONFIG_CPP11_IS_ENUM -#include -#endif - -namespace Catch { - -// Why we're here. -template -std::string toString( T const& value ); - -// Built in overloads - -std::string toString( std::string const& value ); -std::string toString( std::wstring const& value ); -std::string toString( const char* const value ); -std::string toString( char* const value ); -std::string toString( const wchar_t* const value ); -std::string toString( wchar_t* const value ); -std::string toString( int value ); -std::string toString( unsigned long value ); -std::string toString( unsigned int value ); -std::string toString( const double value ); -std::string toString( const float value ); -std::string toString( bool value ); -std::string toString( char value ); -std::string toString( signed char value ); -std::string toString( unsigned char value ); - -#ifdef CATCH_CONFIG_CPP11_LONG_LONG -std::string toString( long long value ); -std::string toString( unsigned long long value ); -#endif - -#ifdef CATCH_CONFIG_CPP11_NULLPTR -std::string toString( std::nullptr_t ); -#endif - -#ifdef __OBJC__ - std::string toString( NSString const * const& nsstring ); - std::string toString( NSString * CATCH_ARC_STRONG & nsstring ); - std::string toString( NSObject* const& nsObject ); -#endif - -namespace Detail { - - extern const std::string unprintableString; - - #if !defined(CATCH_CONFIG_CPP11_STREAM_INSERTABLE_CHECK) - struct BorgType { - template BorgType( T const& ); - }; - - struct TrueType { char sizer[1]; }; - struct FalseType { char sizer[2]; }; - - TrueType& testStreamable( std::ostream& ); - FalseType testStreamable( FalseType ); - - FalseType operator<<( std::ostream const&, BorgType const& ); - - template - struct IsStreamInsertable { - static std::ostream &s; - static T const&t; - enum { value = sizeof( testStreamable(s << t) ) == sizeof( TrueType ) }; - }; -#else - template - class IsStreamInsertable { - template - static auto test(int) - -> decltype( std::declval() << std::declval(), std::true_type() ); - - template - static auto test(...) -> std::false_type; - - public: - static const bool value = decltype(test(0))::value; - }; -#endif - -#if defined(CATCH_CONFIG_CPP11_IS_ENUM) - template::value - > - struct EnumStringMaker - { - static std::string convert( T const& ) { return unprintableString; } - }; - - template - struct EnumStringMaker - { - static std::string convert( T const& v ) - { - return ::Catch::toString( - static_cast::type>(v) - ); - } - }; -#endif - template - struct StringMakerBase { -#if defined(CATCH_CONFIG_CPP11_IS_ENUM) - template - static std::string convert( T const& v ) - { - return EnumStringMaker::convert( v ); - } -#else - template - static std::string convert( T const& ) { return unprintableString; } -#endif - }; - - template<> - struct StringMakerBase { - template - static std::string convert( T const& _value ) { - std::ostringstream oss; - oss << _value; - return oss.str(); - } - }; - - std::string rawMemoryToString( const void *object, std::size_t size ); - - template - std::string rawMemoryToString( const T& object ) { - return rawMemoryToString( &object, sizeof(object) ); - } - -} // end namespace Detail - -template -struct StringMaker : - Detail::StringMakerBase::value> {}; - -template -struct StringMaker { - template - static std::string convert( U* p ) { - if( !p ) - return "NULL"; - else - return Detail::rawMemoryToString( p ); - } -}; - -template -struct StringMaker { - static std::string convert( R C::* p ) { - if( !p ) - return "NULL"; - else - return Detail::rawMemoryToString( p ); - } -}; - -namespace Detail { - template - std::string rangeToString( InputIterator first, InputIterator last ); -} - -//template -//struct StringMaker > { -// static std::string convert( std::vector const& v ) { -// return Detail::rangeToString( v.begin(), v.end() ); -// } -//}; - -template -std::string toString( std::vector const& v ) { - return Detail::rangeToString( v.begin(), v.end() ); -} - -#ifdef CATCH_CONFIG_CPP11_TUPLE - -// toString for tuples -namespace TupleDetail { - template< - typename Tuple, - std::size_t N = 0, - bool = (N < std::tuple_size::value) - > - struct ElementPrinter { - static void print( const Tuple& tuple, std::ostream& os ) - { - os << ( N ? ", " : " " ) - << Catch::toString(std::get(tuple)); - ElementPrinter::print(tuple,os); - } - }; - - template< - typename Tuple, - std::size_t N - > - struct ElementPrinter { - static void print( const Tuple&, std::ostream& ) {} - }; - -} - -template -struct StringMaker> { - - static std::string convert( const std::tuple& tuple ) - { - std::ostringstream os; - os << '{'; - TupleDetail::ElementPrinter>::print( tuple, os ); - os << " }"; - return os.str(); - } -}; -#endif // CATCH_CONFIG_CPP11_TUPLE - -namespace Detail { - template - std::string makeString( T const& value ) { - return StringMaker::convert( value ); - } -} // end namespace Detail - -/// \brief converts any type to a string -/// -/// The default template forwards on to ostringstream - except when an -/// ostringstream overload does not exist - in which case it attempts to detect -/// that and writes {?}. -/// Overload (not specialise) this template for custom typs that you don't want -/// to provide an ostream overload for. -template -std::string toString( T const& value ) { - return StringMaker::convert( value ); -} - - namespace Detail { - template - std::string rangeToString( InputIterator first, InputIterator last ) { - std::ostringstream oss; - oss << "{ "; - if( first != last ) { - oss << Catch::toString( *first ); - for( ++first ; first != last ; ++first ) - oss << ", " << Catch::toString( *first ); - } - oss << " }"; - return oss.str(); - } -} - -} // end namespace Catch - -namespace Catch { - -template -class BinaryExpression; - -template -class MatchExpression; - -// Wraps the LHS of an expression and overloads comparison operators -// for also capturing those and RHS (if any) -template -class ExpressionLhs : public DecomposedExpression { -public: - ExpressionLhs( ResultBuilder& rb, T lhs ) : m_rb( rb ), m_lhs( lhs ), m_truthy(false) {} - - ExpressionLhs& operator = ( const ExpressionLhs& ); - - template - BinaryExpression - operator == ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - template - BinaryExpression - operator != ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - template - BinaryExpression - operator < ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - template - BinaryExpression - operator > ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - template - BinaryExpression - operator <= ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - template - BinaryExpression - operator >= ( RhsT const& rhs ) { - return captureExpression( rhs ); - } - - BinaryExpression operator == ( bool rhs ) { - return captureExpression( rhs ); - } - - BinaryExpression operator != ( bool rhs ) { - return captureExpression( rhs ); - } - - void endExpression() { - m_truthy = m_lhs ? true : false; - m_rb - .setResultType( m_truthy ) - .endExpression( *this ); - } - - virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE { - dest = Catch::toString( m_lhs ); - } - -private: - template - BinaryExpression captureExpression( RhsT& rhs ) const { - return BinaryExpression( m_rb, m_lhs, rhs ); - } - - template - BinaryExpression captureExpression( bool rhs ) const { - return BinaryExpression( m_rb, m_lhs, rhs ); - } - -private: - ResultBuilder& m_rb; - T m_lhs; - bool m_truthy; -}; - -template -class BinaryExpression : public DecomposedExpression { -public: - BinaryExpression( ResultBuilder& rb, LhsT lhs, RhsT rhs ) - : m_rb( rb ), m_lhs( lhs ), m_rhs( rhs ) {} - - BinaryExpression& operator = ( BinaryExpression& ); - - void endExpression() const { - m_rb - .setResultType( Internal::compare( m_lhs, m_rhs ) ) - .endExpression( *this ); - } - - virtual bool isBinaryExpression() const CATCH_OVERRIDE { - return true; - } - - virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE { - std::string lhs = Catch::toString( m_lhs ); - std::string rhs = Catch::toString( m_rhs ); - char delim = lhs.size() + rhs.size() < 40 && - lhs.find('\n') == std::string::npos && - rhs.find('\n') == std::string::npos ? ' ' : '\n'; - dest.reserve( 7 + lhs.size() + rhs.size() ); - // 2 for spaces around operator - // 2 for operator - // 2 for parentheses (conditionally added later) - // 1 for negation (conditionally added later) - dest = lhs; - dest += delim; - dest += Internal::OperatorTraits::getName(); - dest += delim; - dest += rhs; - } - -private: - ResultBuilder& m_rb; - LhsT m_lhs; - RhsT m_rhs; -}; - -template -class MatchExpression : public DecomposedExpression { -public: - MatchExpression( ArgT arg, MatcherT matcher, char const* matcherString ) - : m_arg( arg ), m_matcher( matcher ), m_matcherString( matcherString ) {} - - virtual bool isBinaryExpression() const CATCH_OVERRIDE { - return true; - } - - virtual void reconstructExpression( std::string& dest ) const CATCH_OVERRIDE { - std::string matcherAsString = m_matcher.toString(); - dest = Catch::toString( m_arg ); - dest += ' '; - if( matcherAsString == Detail::unprintableString ) - dest += m_matcherString; - else - dest += matcherAsString; - } - -private: - ArgT m_arg; - MatcherT m_matcher; - char const* m_matcherString; -}; - -} // end namespace Catch - - -namespace Catch { - - template - ExpressionLhs ResultBuilder::operator <= ( T const& operand ) { - return ExpressionLhs( *this, operand ); - } - - inline ExpressionLhs ResultBuilder::operator <= ( bool value ) { - return ExpressionLhs( *this, value ); - } - - template - void ResultBuilder::captureMatch( ArgT const& arg, MatcherT const& matcher, - char const* matcherString ) { - MatchExpression expr( arg, matcher, matcherString ); - setResultType( matcher.match( arg ) ); - endExpression( expr ); - } - -} // namespace Catch - -// #included from: catch_message.h -#define TWOBLUECUBES_CATCH_MESSAGE_H_INCLUDED - -#include - -namespace Catch { - - struct MessageInfo { - MessageInfo( std::string const& _macroName, - SourceLineInfo const& _lineInfo, - ResultWas::OfType _type ); - - std::string macroName; - SourceLineInfo lineInfo; - ResultWas::OfType type; - std::string message; - unsigned int sequence; - - bool operator == ( MessageInfo const& other ) const { - return sequence == other.sequence; - } - bool operator < ( MessageInfo const& other ) const { - return sequence < other.sequence; - } - private: - static unsigned int globalCount; - }; - - struct MessageBuilder { - MessageBuilder( std::string const& macroName, - SourceLineInfo const& lineInfo, - ResultWas::OfType type ) - : m_info( macroName, lineInfo, type ) - {} - - template - MessageBuilder& operator << ( T const& value ) { - m_stream << value; - return *this; - } - - MessageInfo m_info; - std::ostringstream m_stream; - }; - - class ScopedMessage { - public: - ScopedMessage( MessageBuilder const& builder ); - ScopedMessage( ScopedMessage const& other ); - ~ScopedMessage(); - - MessageInfo m_info; - }; - -} // end namespace Catch - -// #included from: catch_interfaces_capture.h -#define TWOBLUECUBES_CATCH_INTERFACES_CAPTURE_H_INCLUDED - -#include - -namespace Catch { - - class TestCase; - class AssertionResult; - struct AssertionInfo; - struct SectionInfo; - struct SectionEndInfo; - struct MessageInfo; - class ScopedMessageBuilder; - struct Counts; - - struct IResultCapture { - - virtual ~IResultCapture(); - - virtual void assertionEnded( AssertionResult const& result ) = 0; - virtual bool sectionStarted( SectionInfo const& sectionInfo, - Counts& assertions ) = 0; - virtual void sectionEnded( SectionEndInfo const& endInfo ) = 0; - virtual void sectionEndedEarly( SectionEndInfo const& endInfo ) = 0; - virtual void pushScopedMessage( MessageInfo const& message ) = 0; - virtual void popScopedMessage( MessageInfo const& message ) = 0; - - virtual std::string getCurrentTestName() const = 0; - virtual const AssertionResult* getLastResult() const = 0; - - virtual void exceptionEarlyReported() = 0; - - virtual void handleFatalErrorCondition( std::string const& message ) = 0; - - virtual bool lastAssertionPassed() = 0; - virtual void assertionPassed() = 0; - virtual void assertionRun() = 0; - }; - - IResultCapture& getResultCapture(); -} - -// #included from: catch_debugger.h -#define TWOBLUECUBES_CATCH_DEBUGGER_H_INCLUDED - -// #included from: catch_platform.h -#define TWOBLUECUBES_CATCH_PLATFORM_H_INCLUDED - -#if defined(__MAC_OS_X_VERSION_MIN_REQUIRED) -# define CATCH_PLATFORM_MAC -#elif defined(__IPHONE_OS_VERSION_MIN_REQUIRED) -# define CATCH_PLATFORM_IPHONE -#elif defined(linux) || defined(__linux) || defined(__linux__) -# define CATCH_PLATFORM_LINUX -#elif defined(WIN32) || defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER) -# define CATCH_PLATFORM_WINDOWS -# if !defined(NOMINMAX) && !defined(CATCH_CONFIG_NO_NOMINMAX) -# define CATCH_DEFINES_NOMINMAX -# endif -# if !defined(WIN32_LEAN_AND_MEAN) && !defined(CATCH_CONFIG_NO_WIN32_LEAN_AND_MEAN) -# define CATCH_DEFINES_WIN32_LEAN_AND_MEAN -# endif -#endif - -#include - -namespace Catch{ - - bool isDebuggerActive(); - void writeToDebugConsole( std::string const& text ); -} - -#ifdef CATCH_PLATFORM_MAC - - // The following code snippet based on: - // http://cocoawithlove.com/2008/03/break-into-debugger.html - #if defined(__ppc64__) || defined(__ppc__) - #define CATCH_TRAP() \ - __asm__("li r0, 20\nsc\nnop\nli r0, 37\nli r4, 2\nsc\nnop\n" \ - : : : "memory","r0","r3","r4" ) /* NOLINT */ - #else - #define CATCH_TRAP() __asm__("int $3\n" : : /* NOLINT */ ) - #endif - -#elif defined(CATCH_PLATFORM_LINUX) - // If we can use inline assembler, do it because this allows us to break - // directly at the location of the failing check instead of breaking inside - // raise() called from it, i.e. one stack frame below. - #if defined(__GNUC__) && (defined(__i386) || defined(__x86_64)) - #define CATCH_TRAP() asm volatile ("int $3") /* NOLINT */ - #else // Fall back to the generic way. - #include - - #define CATCH_TRAP() raise(SIGTRAP) - #endif -#elif defined(_MSC_VER) - #define CATCH_TRAP() __debugbreak() -#elif defined(__MINGW32__) - extern "C" __declspec(dllimport) void __stdcall DebugBreak(); - #define CATCH_TRAP() DebugBreak() -#endif - -#ifdef CATCH_TRAP - #define CATCH_BREAK_INTO_DEBUGGER() if( Catch::isDebuggerActive() ) { CATCH_TRAP(); } -#else - #define CATCH_BREAK_INTO_DEBUGGER() Catch::alwaysTrue(); -#endif - -// #included from: catch_interfaces_runner.h -#define TWOBLUECUBES_CATCH_INTERFACES_RUNNER_H_INCLUDED - -namespace Catch { - class TestCase; - - struct IRunner { - virtual ~IRunner(); - virtual bool aborting() const = 0; - }; -} - -#if !defined(CATCH_CONFIG_DISABLE_STRINGIFICATION) -# define CATCH_INTERNAL_STRINGIFY(expr) #expr -#else -# define CATCH_INTERNAL_STRINGIFY(expr) "Disabled by CATCH_CONFIG_DISABLE_STRINGIFICATION" -#endif - -#if defined(CATCH_CONFIG_FAST_COMPILE) -/////////////////////////////////////////////////////////////////////////////// -// We can speedup compilation significantly by breaking into debugger lower in -// the callstack, because then we don't have to expand CATCH_BREAK_INTO_DEBUGGER -// macro in each assertion -#define INTERNAL_CATCH_REACT( resultBuilder ) \ - resultBuilder.react(); - -/////////////////////////////////////////////////////////////////////////////// -// Another way to speed-up compilation is to omit local try-catch for REQUIRE* -// macros. -// This can potentially cause false negative, if the test code catches -// the exception before it propagates back up to the runner. -#define INTERNAL_CATCH_TEST_NO_TRY( macroName, resultDisposition, expr ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(expr), resultDisposition ); \ - __catchResult.setExceptionGuard(); \ - CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ - ( __catchResult <= expr ).endExpression(); \ - CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ - __catchResult.unsetExceptionGuard(); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::isTrue( false && static_cast( !!(expr) ) ) ) // expr here is never evaluated at runtime but it forces the compiler to give it a look -// The double negation silences MSVC's C4800 warning, the static_cast forces short-circuit evaluation if the type has overloaded &&. - -#define INTERNAL_CHECK_THAT_NO_TRY( macroName, matcher, resultDisposition, arg ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(arg) ", " CATCH_INTERNAL_STRINGIFY(matcher), resultDisposition ); \ - __catchResult.setExceptionGuard(); \ - __catchResult.captureMatch( arg, matcher, CATCH_INTERNAL_STRINGIFY(matcher) ); \ - __catchResult.unsetExceptionGuard(); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) - -#else -/////////////////////////////////////////////////////////////////////////////// -// In the event of a failure works out if the debugger needs to be invoked -// and/or an exception thrown and takes appropriate action. -// This needs to be done as a macro so the debugger will stop in the user -// source code rather than in Catch library code -#define INTERNAL_CATCH_REACT( resultBuilder ) \ - if( resultBuilder.shouldDebugBreak() ) CATCH_BREAK_INTO_DEBUGGER(); \ - resultBuilder.react(); -#endif - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(expr), resultDisposition ); \ - try { \ - CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ - ( __catchResult <= expr ).endExpression(); \ - CATCH_INTERNAL_UNSUPPRESS_PARENTHESES_WARNINGS \ - } \ - catch( ... ) { \ - __catchResult.useActiveException( resultDisposition ); \ - } \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::isTrue( false && static_cast( !!(expr) ) ) ) // expr here is never evaluated at runtime but it forces the compiler to give it a look - // The double negation silences MSVC's C4800 warning, the static_cast forces short-circuit evaluation if the type has overloaded &&. - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_IF( macroName, resultDisposition, expr ) \ - INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ); \ - if( Catch::getResultCapture().lastAssertionPassed() ) - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_ELSE( macroName, resultDisposition, expr ) \ - INTERNAL_CATCH_TEST( macroName, resultDisposition, expr ); \ - if( !Catch::getResultCapture().lastAssertionPassed() ) - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_NO_THROW( macroName, resultDisposition, expr ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(expr), resultDisposition ); \ - try { \ - static_cast(expr); \ - __catchResult.captureResult( Catch::ResultWas::Ok ); \ - } \ - catch( ... ) { \ - __catchResult.useActiveException( resultDisposition ); \ - } \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_THROWS( macroName, resultDisposition, matcher, expr ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(expr), resultDisposition, CATCH_INTERNAL_STRINGIFY(matcher) ); \ - if( __catchResult.allowThrows() ) \ - try { \ - static_cast(expr); \ - __catchResult.captureResult( Catch::ResultWas::DidntThrowException ); \ - } \ - catch( ... ) { \ - __catchResult.captureExpectedException( matcher ); \ - } \ - else \ - __catchResult.captureResult( Catch::ResultWas::Ok ); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_THROWS_AS( macroName, exceptionType, resultDisposition, expr ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(expr) ", " CATCH_INTERNAL_STRINGIFY(exceptionType), resultDisposition ); \ - if( __catchResult.allowThrows() ) \ - try { \ - static_cast(expr); \ - __catchResult.captureResult( Catch::ResultWas::DidntThrowException ); \ - } \ - catch( exceptionType ) { \ - __catchResult.captureResult( Catch::ResultWas::Ok ); \ - } \ - catch( ... ) { \ - __catchResult.useActiveException( resultDisposition ); \ - } \ - else \ - __catchResult.captureResult( Catch::ResultWas::Ok ); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) - -/////////////////////////////////////////////////////////////////////////////// -#ifdef CATCH_CONFIG_VARIADIC_MACROS - #define INTERNAL_CATCH_MSG( macroName, messageType, resultDisposition, ... ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, "", resultDisposition ); \ - __catchResult << __VA_ARGS__ + ::Catch::StreamEndStop(); \ - __catchResult.captureResult( messageType ); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) -#else - #define INTERNAL_CATCH_MSG( macroName, messageType, resultDisposition, log ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, "", resultDisposition ); \ - __catchResult << log + ::Catch::StreamEndStop(); \ - __catchResult.captureResult( messageType ); \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) -#endif - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_INFO( macroName, log ) \ - Catch::ScopedMessage INTERNAL_CATCH_UNIQUE_NAME( scopedMessage ) = Catch::MessageBuilder( macroName, CATCH_INTERNAL_LINEINFO, Catch::ResultWas::Info ) << log; - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CHECK_THAT( macroName, matcher, resultDisposition, arg ) \ - do { \ - Catch::ResultBuilder __catchResult( macroName, CATCH_INTERNAL_LINEINFO, CATCH_INTERNAL_STRINGIFY(arg) ", " CATCH_INTERNAL_STRINGIFY(matcher), resultDisposition ); \ - try { \ - __catchResult.captureMatch( arg, matcher, CATCH_INTERNAL_STRINGIFY(matcher) ); \ - } catch( ... ) { \ - __catchResult.useActiveException( resultDisposition | Catch::ResultDisposition::ContinueOnFailure ); \ - } \ - INTERNAL_CATCH_REACT( __catchResult ) \ - } while( Catch::alwaysFalse() ) - -// #included from: internal/catch_section.h -#define TWOBLUECUBES_CATCH_SECTION_H_INCLUDED - -// #included from: catch_section_info.h -#define TWOBLUECUBES_CATCH_SECTION_INFO_H_INCLUDED - -// #included from: catch_totals.hpp -#define TWOBLUECUBES_CATCH_TOTALS_HPP_INCLUDED - -#include - -namespace Catch { - - struct Counts { - Counts() : passed( 0 ), failed( 0 ), failedButOk( 0 ) {} - - Counts operator - ( Counts const& other ) const { - Counts diff; - diff.passed = passed - other.passed; - diff.failed = failed - other.failed; - diff.failedButOk = failedButOk - other.failedButOk; - return diff; - } - Counts& operator += ( Counts const& other ) { - passed += other.passed; - failed += other.failed; - failedButOk += other.failedButOk; - return *this; - } - - std::size_t total() const { - return passed + failed + failedButOk; - } - bool allPassed() const { - return failed == 0 && failedButOk == 0; - } - bool allOk() const { - return failed == 0; - } - - std::size_t passed; - std::size_t failed; - std::size_t failedButOk; - }; - - struct Totals { - - Totals operator - ( Totals const& other ) const { - Totals diff; - diff.assertions = assertions - other.assertions; - diff.testCases = testCases - other.testCases; - return diff; - } - - Totals delta( Totals const& prevTotals ) const { - Totals diff = *this - prevTotals; - if( diff.assertions.failed > 0 ) - ++diff.testCases.failed; - else if( diff.assertions.failedButOk > 0 ) - ++diff.testCases.failedButOk; - else - ++diff.testCases.passed; - return diff; - } - - Totals& operator += ( Totals const& other ) { - assertions += other.assertions; - testCases += other.testCases; - return *this; - } - - Counts assertions; - Counts testCases; - }; -} - -#include - -namespace Catch { - - struct SectionInfo { - SectionInfo - ( SourceLineInfo const& _lineInfo, - std::string const& _name, - std::string const& _description = std::string() ); - - std::string name; - std::string description; - SourceLineInfo lineInfo; - }; - - struct SectionEndInfo { - SectionEndInfo( SectionInfo const& _sectionInfo, Counts const& _prevAssertions, double _durationInSeconds ) - : sectionInfo( _sectionInfo ), prevAssertions( _prevAssertions ), durationInSeconds( _durationInSeconds ) - {} - - SectionInfo sectionInfo; - Counts prevAssertions; - double durationInSeconds; - }; - -} // end namespace Catch - -// #included from: catch_timer.h -#define TWOBLUECUBES_CATCH_TIMER_H_INCLUDED - -#ifdef _MSC_VER - -namespace Catch { - typedef unsigned long long UInt64; -} -#else -#include -namespace Catch { - typedef uint64_t UInt64; -} -#endif - -namespace Catch { - class Timer { - public: - Timer() : m_ticks( 0 ) {} - void start(); - unsigned int getElapsedMicroseconds() const; - unsigned int getElapsedMilliseconds() const; - double getElapsedSeconds() const; - - private: - UInt64 m_ticks; - }; - -} // namespace Catch - -#include - -namespace Catch { - - class Section : NonCopyable { - public: - Section( SectionInfo const& info ); - ~Section(); - - // This indicates whether the section should be executed or not - operator bool() const; - - private: - SectionInfo m_info; - - std::string m_name; - Counts m_assertions; - bool m_sectionIncluded; - Timer m_timer; - }; - -} // end namespace Catch - -#ifdef CATCH_CONFIG_VARIADIC_MACROS - #define INTERNAL_CATCH_SECTION( ... ) \ - if( Catch::Section const& INTERNAL_CATCH_UNIQUE_NAME( catch_internal_Section ) = Catch::SectionInfo( CATCH_INTERNAL_LINEINFO, __VA_ARGS__ ) ) -#else - #define INTERNAL_CATCH_SECTION( name, desc ) \ - if( Catch::Section const& INTERNAL_CATCH_UNIQUE_NAME( catch_internal_Section ) = Catch::SectionInfo( CATCH_INTERNAL_LINEINFO, name, desc ) ) -#endif - -// #included from: internal/catch_generators.hpp -#define TWOBLUECUBES_CATCH_GENERATORS_HPP_INCLUDED - -#include -#include -#include - -namespace Catch { - -template -struct IGenerator { - virtual ~IGenerator() {} - virtual T getValue( std::size_t index ) const = 0; - virtual std::size_t size () const = 0; -}; - -template -class BetweenGenerator : public IGenerator { -public: - BetweenGenerator( T from, T to ) : m_from( from ), m_to( to ){} - - virtual T getValue( std::size_t index ) const { - return m_from+static_cast( index ); - } - - virtual std::size_t size() const { - return static_cast( 1+m_to-m_from ); - } - -private: - - T m_from; - T m_to; -}; - -template -class ValuesGenerator : public IGenerator { -public: - ValuesGenerator(){} - - void add( T value ) { - m_values.push_back( value ); - } - - virtual T getValue( std::size_t index ) const { - return m_values[index]; - } - - virtual std::size_t size() const { - return m_values.size(); - } - -private: - std::vector m_values; -}; - -template -class CompositeGenerator { -public: - CompositeGenerator() : m_totalSize( 0 ) {} - - // *** Move semantics, similar to auto_ptr *** - CompositeGenerator( CompositeGenerator& other ) - : m_fileInfo( other.m_fileInfo ), - m_totalSize( 0 ) - { - move( other ); - } - - CompositeGenerator& setFileInfo( const char* fileInfo ) { - m_fileInfo = fileInfo; - return *this; - } - - ~CompositeGenerator() { - deleteAll( m_composed ); - } - - operator T () const { - size_t overallIndex = getCurrentContext().getGeneratorIndex( m_fileInfo, m_totalSize ); - - typename std::vector*>::const_iterator it = m_composed.begin(); - typename std::vector*>::const_iterator itEnd = m_composed.end(); - for( size_t index = 0; it != itEnd; ++it ) - { - const IGenerator* generator = *it; - if( overallIndex >= index && overallIndex < index + generator->size() ) - { - return generator->getValue( overallIndex-index ); - } - index += generator->size(); - } - CATCH_INTERNAL_ERROR( "Indexed past end of generated range" ); - return T(); // Suppress spurious "not all control paths return a value" warning in Visual Studio - if you know how to fix this please do so - } - - void add( const IGenerator* generator ) { - m_totalSize += generator->size(); - m_composed.push_back( generator ); - } - - CompositeGenerator& then( CompositeGenerator& other ) { - move( other ); - return *this; - } - - CompositeGenerator& then( T value ) { - ValuesGenerator* valuesGen = new ValuesGenerator(); - valuesGen->add( value ); - add( valuesGen ); - return *this; - } - -private: - - void move( CompositeGenerator& other ) { - m_composed.insert( m_composed.end(), other.m_composed.begin(), other.m_composed.end() ); - m_totalSize += other.m_totalSize; - other.m_composed.clear(); - } - - std::vector*> m_composed; - std::string m_fileInfo; - size_t m_totalSize; -}; - -namespace Generators -{ - template - CompositeGenerator between( T from, T to ) { - CompositeGenerator generators; - generators.add( new BetweenGenerator( from, to ) ); - return generators; - } - - template - CompositeGenerator values( T val1, T val2 ) { - CompositeGenerator generators; - ValuesGenerator* valuesGen = new ValuesGenerator(); - valuesGen->add( val1 ); - valuesGen->add( val2 ); - generators.add( valuesGen ); - return generators; - } - - template - CompositeGenerator values( T val1, T val2, T val3 ){ - CompositeGenerator generators; - ValuesGenerator* valuesGen = new ValuesGenerator(); - valuesGen->add( val1 ); - valuesGen->add( val2 ); - valuesGen->add( val3 ); - generators.add( valuesGen ); - return generators; - } - - template - CompositeGenerator values( T val1, T val2, T val3, T val4 ) { - CompositeGenerator generators; - ValuesGenerator* valuesGen = new ValuesGenerator(); - valuesGen->add( val1 ); - valuesGen->add( val2 ); - valuesGen->add( val3 ); - valuesGen->add( val4 ); - generators.add( valuesGen ); - return generators; - } - -} // end namespace Generators - -using namespace Generators; - -} // end namespace Catch - -#define INTERNAL_CATCH_LINESTR2( line ) #line -#define INTERNAL_CATCH_LINESTR( line ) INTERNAL_CATCH_LINESTR2( line ) - -#define INTERNAL_CATCH_GENERATE( expr ) expr.setFileInfo( __FILE__ "(" INTERNAL_CATCH_LINESTR( __LINE__ ) ")" ) - -// #included from: internal/catch_interfaces_exception.h -#define TWOBLUECUBES_CATCH_INTERFACES_EXCEPTION_H_INCLUDED - -#include -#include - -// #included from: catch_interfaces_registry_hub.h -#define TWOBLUECUBES_CATCH_INTERFACES_REGISTRY_HUB_H_INCLUDED - -#include - -namespace Catch { - - class TestCase; - struct ITestCaseRegistry; - struct IExceptionTranslatorRegistry; - struct IExceptionTranslator; - struct IReporterRegistry; - struct IReporterFactory; - struct ITagAliasRegistry; - - struct IRegistryHub { - virtual ~IRegistryHub(); - - virtual IReporterRegistry const& getReporterRegistry() const = 0; - virtual ITestCaseRegistry const& getTestCaseRegistry() const = 0; - virtual ITagAliasRegistry const& getTagAliasRegistry() const = 0; - - virtual IExceptionTranslatorRegistry& getExceptionTranslatorRegistry() = 0; - }; - - struct IMutableRegistryHub { - virtual ~IMutableRegistryHub(); - virtual void registerReporter( std::string const& name, Ptr const& factory ) = 0; - virtual void registerListener( Ptr const& factory ) = 0; - virtual void registerTest( TestCase const& testInfo ) = 0; - virtual void registerTranslator( const IExceptionTranslator* translator ) = 0; - virtual void registerTagAlias( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) = 0; - }; - - IRegistryHub& getRegistryHub(); - IMutableRegistryHub& getMutableRegistryHub(); - void cleanUp(); - std::string translateActiveException(); - -} - -namespace Catch { - - typedef std::string(*exceptionTranslateFunction)(); - - struct IExceptionTranslator; - typedef std::vector ExceptionTranslators; - - struct IExceptionTranslator { - virtual ~IExceptionTranslator(); - virtual std::string translate( ExceptionTranslators::const_iterator it, ExceptionTranslators::const_iterator itEnd ) const = 0; - }; - - struct IExceptionTranslatorRegistry { - virtual ~IExceptionTranslatorRegistry(); - - virtual std::string translateActiveException() const = 0; - }; - - class ExceptionTranslatorRegistrar { - template - class ExceptionTranslator : public IExceptionTranslator { - public: - - ExceptionTranslator( std::string(*translateFunction)( T& ) ) - : m_translateFunction( translateFunction ) - {} - - virtual std::string translate( ExceptionTranslators::const_iterator it, ExceptionTranslators::const_iterator itEnd ) const CATCH_OVERRIDE { - try { - if( it == itEnd ) - throw; - else - return (*it)->translate( it+1, itEnd ); - } - catch( T& ex ) { - return m_translateFunction( ex ); - } - } - - protected: - std::string(*m_translateFunction)( T& ); - }; - - public: - template - ExceptionTranslatorRegistrar( std::string(*translateFunction)( T& ) ) { - getMutableRegistryHub().registerTranslator - ( new ExceptionTranslator( translateFunction ) ); - } - }; -} - -/////////////////////////////////////////////////////////////////////////////// -#define INTERNAL_CATCH_TRANSLATE_EXCEPTION2( translatorName, signature ) \ - static std::string translatorName( signature ); \ - namespace{ Catch::ExceptionTranslatorRegistrar INTERNAL_CATCH_UNIQUE_NAME( catch_internal_ExceptionRegistrar )( &translatorName ); }\ - static std::string translatorName( signature ) - -#define INTERNAL_CATCH_TRANSLATE_EXCEPTION( signature ) INTERNAL_CATCH_TRANSLATE_EXCEPTION2( INTERNAL_CATCH_UNIQUE_NAME( catch_internal_ExceptionTranslator ), signature ) - -// #included from: internal/catch_approx.hpp -#define TWOBLUECUBES_CATCH_APPROX_HPP_INCLUDED - -#include -#include - -#if defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) -#include -#endif - -namespace Catch { -namespace Detail { - - class Approx { - public: - explicit Approx ( double value ) - : m_epsilon( std::numeric_limits::epsilon()*100 ), - m_margin( 0.0 ), - m_scale( 1.0 ), - m_value( value ) - {} - - static Approx custom() { - return Approx( 0 ); - } - -#if defined(CATCH_CONFIG_CPP11_TYPE_TRAITS) - - template ::value>::type> - Approx operator()( T value ) { - Approx approx( static_cast(value) ); - approx.epsilon( m_epsilon ); - approx.margin( m_margin ); - approx.scale( m_scale ); - return approx; - } - - template ::value>::type> - explicit Approx( T value ): Approx(static_cast(value)) - {} - - template ::value>::type> - friend bool operator == ( const T& lhs, Approx const& rhs ) { - // Thanks to Richard Harris for his help refining this formula - auto lhs_v = double(lhs); - bool relativeOK = std::fabs(lhs_v - rhs.m_value) < rhs.m_epsilon * (rhs.m_scale + (std::max)(std::fabs(lhs_v), std::fabs(rhs.m_value))); - if (relativeOK) { - return true; - } - - return std::fabs(lhs_v - rhs.m_value) <= rhs.m_margin; - } - - template ::value>::type> - friend bool operator == ( Approx const& lhs, const T& rhs ) { - return operator==( rhs, lhs ); - } - - template ::value>::type> - friend bool operator != ( T lhs, Approx const& rhs ) { - return !operator==( lhs, rhs ); - } - - template ::value>::type> - friend bool operator != ( Approx const& lhs, T rhs ) { - return !operator==( rhs, lhs ); - } - - template ::value>::type> - friend bool operator <= ( T lhs, Approx const& rhs ) { - return double(lhs) < rhs.m_value || lhs == rhs; - } - - template ::value>::type> - friend bool operator <= ( Approx const& lhs, T rhs ) { - return lhs.m_value < double(rhs) || lhs == rhs; - } - - template ::value>::type> - friend bool operator >= ( T lhs, Approx const& rhs ) { - return double(lhs) > rhs.m_value || lhs == rhs; - } - - template ::value>::type> - friend bool operator >= ( Approx const& lhs, T rhs ) { - return lhs.m_value > double(rhs) || lhs == rhs; - } - - template ::value>::type> - Approx& epsilon( T newEpsilon ) { - m_epsilon = double(newEpsilon); - return *this; - } - - template ::value>::type> - Approx& margin( T newMargin ) { - m_margin = double(newMargin); - return *this; - } - - template ::value>::type> - Approx& scale( T newScale ) { - m_scale = double(newScale); - return *this; - } - -#else - - Approx operator()( double value ) { - Approx approx( value ); - approx.epsilon( m_epsilon ); - approx.margin( m_margin ); - approx.scale( m_scale ); - return approx; - } - - friend bool operator == ( double lhs, Approx const& rhs ) { - // Thanks to Richard Harris for his help refining this formula - bool relativeOK = std::fabs( lhs - rhs.m_value ) < rhs.m_epsilon * (rhs.m_scale + (std::max)( std::fabs(lhs), std::fabs(rhs.m_value) ) ); - if (relativeOK) { - return true; - } - return std::fabs(lhs - rhs.m_value) <= rhs.m_margin; - } - - friend bool operator == ( Approx const& lhs, double rhs ) { - return operator==( rhs, lhs ); - } - - friend bool operator != ( double lhs, Approx const& rhs ) { - return !operator==( lhs, rhs ); - } - - friend bool operator != ( Approx const& lhs, double rhs ) { - return !operator==( rhs, lhs ); - } - - friend bool operator <= ( double lhs, Approx const& rhs ) { - return lhs < rhs.m_value || lhs == rhs; - } - - friend bool operator <= ( Approx const& lhs, double rhs ) { - return lhs.m_value < rhs || lhs == rhs; - } - - friend bool operator >= ( double lhs, Approx const& rhs ) { - return lhs > rhs.m_value || lhs == rhs; - } - - friend bool operator >= ( Approx const& lhs, double rhs ) { - return lhs.m_value > rhs || lhs == rhs; - } - - Approx& epsilon( double newEpsilon ) { - m_epsilon = newEpsilon; - return *this; - } - - Approx& margin( double newMargin ) { - m_margin = newMargin; - return *this; - } - - Approx& scale( double newScale ) { - m_scale = newScale; - return *this; - } -#endif - - std::string toString() const { - std::ostringstream oss; - oss << "Approx( " << Catch::toString( m_value ) << " )"; - return oss.str(); - } - - private: - double m_epsilon; - double m_margin; - double m_scale; - double m_value; - }; -} - -template<> -inline std::string toString( Detail::Approx const& value ) { - return value.toString(); -} - -} // end namespace Catch - -// #included from: internal/catch_matchers_string.h -#define TWOBLUECUBES_CATCH_MATCHERS_STRING_H_INCLUDED - -namespace Catch { -namespace Matchers { - - namespace StdString { - - struct CasedString - { - CasedString( std::string const& str, CaseSensitive::Choice caseSensitivity ); - std::string adjustString( std::string const& str ) const; - std::string caseSensitivitySuffix() const; - - CaseSensitive::Choice m_caseSensitivity; - std::string m_str; - }; - - struct StringMatcherBase : MatcherBase { - StringMatcherBase( std::string const& operation, CasedString const& comparator ); - virtual std::string describe() const CATCH_OVERRIDE; - - CasedString m_comparator; - std::string m_operation; - }; - - struct EqualsMatcher : StringMatcherBase { - EqualsMatcher( CasedString const& comparator ); - virtual bool match( std::string const& source ) const CATCH_OVERRIDE; - }; - struct ContainsMatcher : StringMatcherBase { - ContainsMatcher( CasedString const& comparator ); - virtual bool match( std::string const& source ) const CATCH_OVERRIDE; - }; - struct StartsWithMatcher : StringMatcherBase { - StartsWithMatcher( CasedString const& comparator ); - virtual bool match( std::string const& source ) const CATCH_OVERRIDE; - }; - struct EndsWithMatcher : StringMatcherBase { - EndsWithMatcher( CasedString const& comparator ); - virtual bool match( std::string const& source ) const CATCH_OVERRIDE; - }; - - } // namespace StdString - - // The following functions create the actual matcher objects. - // This allows the types to be inferred - - StdString::EqualsMatcher Equals( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); - StdString::ContainsMatcher Contains( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); - StdString::EndsWithMatcher EndsWith( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); - StdString::StartsWithMatcher StartsWith( std::string const& str, CaseSensitive::Choice caseSensitivity = CaseSensitive::Yes ); - -} // namespace Matchers -} // namespace Catch - -// #included from: internal/catch_matchers_vector.h -#define TWOBLUECUBES_CATCH_MATCHERS_VECTOR_H_INCLUDED - -namespace Catch { -namespace Matchers { - - namespace Vector { - - template - struct ContainsElementMatcher : MatcherBase, T> { - - ContainsElementMatcher(T const &comparator) : m_comparator( comparator) {} - - bool match(std::vector const &v) const CATCH_OVERRIDE { - return std::find(v.begin(), v.end(), m_comparator) != v.end(); - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "Contains: " + Catch::toString( m_comparator ); - } - - T const& m_comparator; - }; - - template - struct ContainsMatcher : MatcherBase, std::vector > { - - ContainsMatcher(std::vector const &comparator) : m_comparator( comparator ) {} - - bool match(std::vector const &v) const CATCH_OVERRIDE { - // !TBD: see note in EqualsMatcher - if (m_comparator.size() > v.size()) - return false; - for (size_t i = 0; i < m_comparator.size(); ++i) - if (std::find(v.begin(), v.end(), m_comparator[i]) == v.end()) - return false; - return true; - } - virtual std::string describe() const CATCH_OVERRIDE { - return "Contains: " + Catch::toString( m_comparator ); - } - - std::vector const& m_comparator; - }; - - template - struct EqualsMatcher : MatcherBase, std::vector > { - - EqualsMatcher(std::vector const &comparator) : m_comparator( comparator ) {} - - bool match(std::vector const &v) const CATCH_OVERRIDE { - // !TBD: This currently works if all elements can be compared using != - // - a more general approach would be via a compare template that defaults - // to using !=. but could be specialised for, e.g. std::vector etc - // - then just call that directly - if (m_comparator.size() != v.size()) - return false; - for (size_t i = 0; i < v.size(); ++i) - if (m_comparator[i] != v[i]) - return false; - return true; - } - virtual std::string describe() const CATCH_OVERRIDE { - return "Equals: " + Catch::toString( m_comparator ); - } - std::vector const& m_comparator; - }; - - } // namespace Vector - - // The following functions create the actual matcher objects. - // This allows the types to be inferred - - template - Vector::ContainsMatcher Contains( std::vector const& comparator ) { - return Vector::ContainsMatcher( comparator ); - } - - template - Vector::ContainsElementMatcher VectorContains( T const& comparator ) { - return Vector::ContainsElementMatcher( comparator ); - } - - template - Vector::EqualsMatcher Equals( std::vector const& comparator ) { - return Vector::EqualsMatcher( comparator ); - } - -} // namespace Matchers -} // namespace Catch - -// #included from: internal/catch_interfaces_tag_alias_registry.h -#define TWOBLUECUBES_CATCH_INTERFACES_TAG_ALIAS_REGISTRY_H_INCLUDED - -// #included from: catch_tag_alias.h -#define TWOBLUECUBES_CATCH_TAG_ALIAS_H_INCLUDED - -#include - -namespace Catch { - - struct TagAlias { - TagAlias( std::string const& _tag, SourceLineInfo _lineInfo ) : tag( _tag ), lineInfo( _lineInfo ) {} - - std::string tag; - SourceLineInfo lineInfo; - }; - - struct RegistrarForTagAliases { - RegistrarForTagAliases( char const* alias, char const* tag, SourceLineInfo const& lineInfo ); - }; - -} // end namespace Catch - -#define CATCH_REGISTER_TAG_ALIAS( alias, spec ) namespace{ Catch::RegistrarForTagAliases INTERNAL_CATCH_UNIQUE_NAME( AutoRegisterTagAlias )( alias, spec, CATCH_INTERNAL_LINEINFO ); } -// #included from: catch_option.hpp -#define TWOBLUECUBES_CATCH_OPTION_HPP_INCLUDED - -namespace Catch { - - // An optional type - template - class Option { - public: - Option() : nullableValue( CATCH_NULL ) {} - Option( T const& _value ) - : nullableValue( new( storage ) T( _value ) ) - {} - Option( Option const& _other ) - : nullableValue( _other ? new( storage ) T( *_other ) : CATCH_NULL ) - {} - - ~Option() { - reset(); - } - - Option& operator= ( Option const& _other ) { - if( &_other != this ) { - reset(); - if( _other ) - nullableValue = new( storage ) T( *_other ); - } - return *this; - } - Option& operator = ( T const& _value ) { - reset(); - nullableValue = new( storage ) T( _value ); - return *this; - } - - void reset() { - if( nullableValue ) - nullableValue->~T(); - nullableValue = CATCH_NULL; - } - - T& operator*() { return *nullableValue; } - T const& operator*() const { return *nullableValue; } - T* operator->() { return nullableValue; } - const T* operator->() const { return nullableValue; } - - T valueOr( T const& defaultValue ) const { - return nullableValue ? *nullableValue : defaultValue; - } - - bool some() const { return nullableValue != CATCH_NULL; } - bool none() const { return nullableValue == CATCH_NULL; } - - bool operator !() const { return nullableValue == CATCH_NULL; } - operator SafeBool::type() const { - return SafeBool::makeSafe( some() ); - } - - private: - T *nullableValue; - union { - char storage[sizeof(T)]; - - // These are here to force alignment for the storage - long double dummy1; - void (*dummy2)(); - long double dummy3; -#ifdef CATCH_CONFIG_CPP11_LONG_LONG - long long dummy4; -#endif - }; - }; - -} // end namespace Catch - -namespace Catch { - - struct ITagAliasRegistry { - virtual ~ITagAliasRegistry(); - virtual Option find( std::string const& alias ) const = 0; - virtual std::string expandAliases( std::string const& unexpandedTestSpec ) const = 0; - - static ITagAliasRegistry const& get(); - }; - -} // end namespace Catch - -// These files are included here so the single_include script doesn't put them -// in the conditionally compiled sections -// #included from: internal/catch_test_case_info.h -#define TWOBLUECUBES_CATCH_TEST_CASE_INFO_H_INCLUDED - -#include -#include - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wpadded" -#endif - -namespace Catch { - - struct ITestCase; - - struct TestCaseInfo { - enum SpecialProperties{ - None = 0, - IsHidden = 1 << 1, - ShouldFail = 1 << 2, - MayFail = 1 << 3, - Throws = 1 << 4, - NonPortable = 1 << 5 - }; - - TestCaseInfo( std::string const& _name, - std::string const& _className, - std::string const& _description, - std::set const& _tags, - SourceLineInfo const& _lineInfo ); - - TestCaseInfo( TestCaseInfo const& other ); - - friend void setTags( TestCaseInfo& testCaseInfo, std::set const& tags ); - - bool isHidden() const; - bool throws() const; - bool okToFail() const; - bool expectedToFail() const; - - std::string name; - std::string className; - std::string description; - std::set tags; - std::set lcaseTags; - std::string tagsAsString; - SourceLineInfo lineInfo; - SpecialProperties properties; - }; - - class TestCase : public TestCaseInfo { - public: - - TestCase( ITestCase* testCase, TestCaseInfo const& info ); - TestCase( TestCase const& other ); - - TestCase withName( std::string const& _newName ) const; - - void invoke() const; - - TestCaseInfo const& getTestCaseInfo() const; - - void swap( TestCase& other ); - bool operator == ( TestCase const& other ) const; - bool operator < ( TestCase const& other ) const; - TestCase& operator = ( TestCase const& other ); - - private: - Ptr test; - }; - - TestCase makeTestCase( ITestCase* testCase, - std::string const& className, - std::string const& name, - std::string const& description, - SourceLineInfo const& lineInfo ); -} - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - - -#ifdef __OBJC__ -// #included from: internal/catch_objc.hpp -#define TWOBLUECUBES_CATCH_OBJC_HPP_INCLUDED - -#import - -#include - -// NB. Any general catch headers included here must be included -// in catch.hpp first to make sure they are included by the single -// header for non obj-usage - -/////////////////////////////////////////////////////////////////////////////// -// This protocol is really only here for (self) documenting purposes, since -// all its methods are optional. -@protocol OcFixture - -@optional - --(void) setUp; --(void) tearDown; - -@end - -namespace Catch { - - class OcMethod : public SharedImpl { - - public: - OcMethod( Class cls, SEL sel ) : m_cls( cls ), m_sel( sel ) {} - - virtual void invoke() const { - id obj = [[m_cls alloc] init]; - - performOptionalSelector( obj, @selector(setUp) ); - performOptionalSelector( obj, m_sel ); - performOptionalSelector( obj, @selector(tearDown) ); - - arcSafeRelease( obj ); - } - private: - virtual ~OcMethod() {} - - Class m_cls; - SEL m_sel; - }; - - namespace Detail{ - - inline std::string getAnnotation( Class cls, - std::string const& annotationName, - std::string const& testCaseName ) { - NSString* selStr = [[NSString alloc] initWithFormat:@"Catch_%s_%s", annotationName.c_str(), testCaseName.c_str()]; - SEL sel = NSSelectorFromString( selStr ); - arcSafeRelease( selStr ); - id value = performOptionalSelector( cls, sel ); - if( value ) - return [(NSString*)value UTF8String]; - return ""; - } - } - - inline size_t registerTestMethods() { - size_t noTestMethods = 0; - int noClasses = objc_getClassList( CATCH_NULL, 0 ); - - Class* classes = (CATCH_UNSAFE_UNRETAINED Class *)malloc( sizeof(Class) * noClasses); - objc_getClassList( classes, noClasses ); - - for( int c = 0; c < noClasses; c++ ) { - Class cls = classes[c]; - { - u_int count; - Method* methods = class_copyMethodList( cls, &count ); - for( u_int m = 0; m < count ; m++ ) { - SEL selector = method_getName(methods[m]); - std::string methodName = sel_getName(selector); - if( startsWith( methodName, "Catch_TestCase_" ) ) { - std::string testCaseName = methodName.substr( 15 ); - std::string name = Detail::getAnnotation( cls, "Name", testCaseName ); - std::string desc = Detail::getAnnotation( cls, "Description", testCaseName ); - const char* className = class_getName( cls ); - - getMutableRegistryHub().registerTest( makeTestCase( new OcMethod( cls, selector ), className, name.c_str(), desc.c_str(), SourceLineInfo() ) ); - noTestMethods++; - } - } - free(methods); - } - } - return noTestMethods; - } - - namespace Matchers { - namespace Impl { - namespace NSStringMatchers { - - struct StringHolder : MatcherBase{ - StringHolder( NSString* substr ) : m_substr( [substr copy] ){} - StringHolder( StringHolder const& other ) : m_substr( [other.m_substr copy] ){} - StringHolder() { - arcSafeRelease( m_substr ); - } - - virtual bool match( NSString* arg ) const CATCH_OVERRIDE { - return false; - } - - NSString* m_substr; - }; - - struct Equals : StringHolder { - Equals( NSString* substr ) : StringHolder( substr ){} - - virtual bool match( NSString* str ) const CATCH_OVERRIDE { - return (str != nil || m_substr == nil ) && - [str isEqualToString:m_substr]; - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "equals string: " + Catch::toString( m_substr ); - } - }; - - struct Contains : StringHolder { - Contains( NSString* substr ) : StringHolder( substr ){} - - virtual bool match( NSString* str ) const { - return (str != nil || m_substr == nil ) && - [str rangeOfString:m_substr].location != NSNotFound; - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "contains string: " + Catch::toString( m_substr ); - } - }; - - struct StartsWith : StringHolder { - StartsWith( NSString* substr ) : StringHolder( substr ){} - - virtual bool match( NSString* str ) const { - return (str != nil || m_substr == nil ) && - [str rangeOfString:m_substr].location == 0; - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "starts with: " + Catch::toString( m_substr ); - } - }; - struct EndsWith : StringHolder { - EndsWith( NSString* substr ) : StringHolder( substr ){} - - virtual bool match( NSString* str ) const { - return (str != nil || m_substr == nil ) && - [str rangeOfString:m_substr].location == [str length] - [m_substr length]; - } - - virtual std::string describe() const CATCH_OVERRIDE { - return "ends with: " + Catch::toString( m_substr ); - } - }; - - } // namespace NSStringMatchers - } // namespace Impl - - inline Impl::NSStringMatchers::Equals - Equals( NSString* substr ){ return Impl::NSStringMatchers::Equals( substr ); } - - inline Impl::NSStringMatchers::Contains - Contains( NSString* substr ){ return Impl::NSStringMatchers::Contains( substr ); } - - inline Impl::NSStringMatchers::StartsWith - StartsWith( NSString* substr ){ return Impl::NSStringMatchers::StartsWith( substr ); } - - inline Impl::NSStringMatchers::EndsWith - EndsWith( NSString* substr ){ return Impl::NSStringMatchers::EndsWith( substr ); } - - } // namespace Matchers - - using namespace Matchers; - -} // namespace Catch - -/////////////////////////////////////////////////////////////////////////////// -#define OC_TEST_CASE( name, desc )\ -+(NSString*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Name_test ) \ -{\ -return @ name; \ -}\ -+(NSString*) INTERNAL_CATCH_UNIQUE_NAME( Catch_Description_test ) \ -{ \ -return @ desc; \ -} \ --(void) INTERNAL_CATCH_UNIQUE_NAME( Catch_TestCase_test ) - -#endif - -#ifdef CATCH_IMPL - -// !TBD: Move the leak detector code into a separate header -#ifdef CATCH_CONFIG_WINDOWS_CRTDBG -#include -class LeakDetector { -public: - LeakDetector() { - int flag = _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG); - flag |= _CRTDBG_LEAK_CHECK_DF; - flag |= _CRTDBG_ALLOC_MEM_DF; - _CrtSetDbgFlag(flag); - _CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE | _CRTDBG_MODE_DEBUG); - _CrtSetReportFile(_CRT_WARN, _CRTDBG_FILE_STDERR); - // Change this to leaking allocation's number to break there - _CrtSetBreakAlloc(-1); - } -}; -#else -class LeakDetector {}; -#endif - -LeakDetector leakDetector; - -// #included from: internal/catch_impl.hpp -#define TWOBLUECUBES_CATCH_IMPL_HPP_INCLUDED - -// Collect all the implementation files together here -// These are the equivalent of what would usually be cpp files - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wweak-vtables" -#endif - -// #included from: ../catch_session.hpp -#define TWOBLUECUBES_CATCH_RUNNER_HPP_INCLUDED - -// #included from: internal/catch_commandline.hpp -#define TWOBLUECUBES_CATCH_COMMANDLINE_HPP_INCLUDED - -// #included from: catch_config.hpp -#define TWOBLUECUBES_CATCH_CONFIG_HPP_INCLUDED - -// #included from: catch_test_spec_parser.hpp -#define TWOBLUECUBES_CATCH_TEST_SPEC_PARSER_HPP_INCLUDED - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wpadded" -#endif - -// #included from: catch_test_spec.hpp -#define TWOBLUECUBES_CATCH_TEST_SPEC_HPP_INCLUDED - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wpadded" -#endif - -// #included from: catch_wildcard_pattern.hpp -#define TWOBLUECUBES_CATCH_WILDCARD_PATTERN_HPP_INCLUDED - -#include - -namespace Catch -{ - class WildcardPattern { - enum WildcardPosition { - NoWildcard = 0, - WildcardAtStart = 1, - WildcardAtEnd = 2, - WildcardAtBothEnds = WildcardAtStart | WildcardAtEnd - }; - - public: - - WildcardPattern( std::string const& pattern, CaseSensitive::Choice caseSensitivity ) - : m_caseSensitivity( caseSensitivity ), - m_wildcard( NoWildcard ), - m_pattern( adjustCase( pattern ) ) - { - if( startsWith( m_pattern, '*' ) ) { - m_pattern = m_pattern.substr( 1 ); - m_wildcard = WildcardAtStart; - } - if( endsWith( m_pattern, '*' ) ) { - m_pattern = m_pattern.substr( 0, m_pattern.size()-1 ); - m_wildcard = static_cast( m_wildcard | WildcardAtEnd ); - } - } - virtual ~WildcardPattern(); - virtual bool matches( std::string const& str ) const { - switch( m_wildcard ) { - case NoWildcard: - return m_pattern == adjustCase( str ); - case WildcardAtStart: - return endsWith( adjustCase( str ), m_pattern ); - case WildcardAtEnd: - return startsWith( adjustCase( str ), m_pattern ); - case WildcardAtBothEnds: - return contains( adjustCase( str ), m_pattern ); - } - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wunreachable-code" -#endif - throw std::logic_error( "Unknown enum" ); -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - } - private: - std::string adjustCase( std::string const& str ) const { - return m_caseSensitivity == CaseSensitive::No ? toLower( str ) : str; - } - CaseSensitive::Choice m_caseSensitivity; - WildcardPosition m_wildcard; - std::string m_pattern; - }; -} - -#include -#include - -namespace Catch { - - class TestSpec { - struct Pattern : SharedImpl<> { - virtual ~Pattern(); - virtual bool matches( TestCaseInfo const& testCase ) const = 0; - }; - class NamePattern : public Pattern { - public: - NamePattern( std::string const& name ) - : m_wildcardPattern( toLower( name ), CaseSensitive::No ) - {} - virtual ~NamePattern(); - virtual bool matches( TestCaseInfo const& testCase ) const { - return m_wildcardPattern.matches( toLower( testCase.name ) ); - } - private: - WildcardPattern m_wildcardPattern; - }; - - class TagPattern : public Pattern { - public: - TagPattern( std::string const& tag ) : m_tag( toLower( tag ) ) {} - virtual ~TagPattern(); - virtual bool matches( TestCaseInfo const& testCase ) const { - return testCase.lcaseTags.find( m_tag ) != testCase.lcaseTags.end(); - } - private: - std::string m_tag; - }; - - class ExcludedPattern : public Pattern { - public: - ExcludedPattern( Ptr const& underlyingPattern ) : m_underlyingPattern( underlyingPattern ) {} - virtual ~ExcludedPattern(); - virtual bool matches( TestCaseInfo const& testCase ) const { return !m_underlyingPattern->matches( testCase ); } - private: - Ptr m_underlyingPattern; - }; - - struct Filter { - std::vector > m_patterns; - - bool matches( TestCaseInfo const& testCase ) const { - // All patterns in a filter must match for the filter to be a match - for( std::vector >::const_iterator it = m_patterns.begin(), itEnd = m_patterns.end(); it != itEnd; ++it ) { - if( !(*it)->matches( testCase ) ) - return false; - } - return true; - } - }; - - public: - bool hasFilters() const { - return !m_filters.empty(); - } - bool matches( TestCaseInfo const& testCase ) const { - // A TestSpec matches if any filter matches - for( std::vector::const_iterator it = m_filters.begin(), itEnd = m_filters.end(); it != itEnd; ++it ) - if( it->matches( testCase ) ) - return true; - return false; - } - - private: - std::vector m_filters; - - friend class TestSpecParser; - }; -} - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - -namespace Catch { - - class TestSpecParser { - enum Mode{ None, Name, QuotedName, Tag, EscapedName }; - Mode m_mode; - bool m_exclusion; - std::size_t m_start, m_pos; - std::string m_arg; - std::vector m_escapeChars; - TestSpec::Filter m_currentFilter; - TestSpec m_testSpec; - ITagAliasRegistry const* m_tagAliases; - - public: - TestSpecParser( ITagAliasRegistry const& tagAliases ) :m_mode(None), m_exclusion(false), m_start(0), m_pos(0), m_tagAliases( &tagAliases ) {} - - TestSpecParser& parse( std::string const& arg ) { - m_mode = None; - m_exclusion = false; - m_start = std::string::npos; - m_arg = m_tagAliases->expandAliases( arg ); - m_escapeChars.clear(); - for( m_pos = 0; m_pos < m_arg.size(); ++m_pos ) - visitChar( m_arg[m_pos] ); - if( m_mode == Name ) - addPattern(); - return *this; - } - TestSpec testSpec() { - addFilter(); - return m_testSpec; - } - private: - void visitChar( char c ) { - if( m_mode == None ) { - switch( c ) { - case ' ': return; - case '~': m_exclusion = true; return; - case '[': return startNewMode( Tag, ++m_pos ); - case '"': return startNewMode( QuotedName, ++m_pos ); - case '\\': return escape(); - default: startNewMode( Name, m_pos ); break; - } - } - if( m_mode == Name ) { - if( c == ',' ) { - addPattern(); - addFilter(); - } - else if( c == '[' ) { - if( subString() == "exclude:" ) - m_exclusion = true; - else - addPattern(); - startNewMode( Tag, ++m_pos ); - } - else if( c == '\\' ) - escape(); - } - else if( m_mode == EscapedName ) - m_mode = Name; - else if( m_mode == QuotedName && c == '"' ) - addPattern(); - else if( m_mode == Tag && c == ']' ) - addPattern(); - } - void startNewMode( Mode mode, std::size_t start ) { - m_mode = mode; - m_start = start; - } - void escape() { - if( m_mode == None ) - m_start = m_pos; - m_mode = EscapedName; - m_escapeChars.push_back( m_pos ); - } - std::string subString() const { return m_arg.substr( m_start, m_pos - m_start ); } - template - void addPattern() { - std::string token = subString(); - for( size_t i = 0; i < m_escapeChars.size(); ++i ) - token = token.substr( 0, m_escapeChars[i]-m_start-i ) + token.substr( m_escapeChars[i]-m_start-i+1 ); - m_escapeChars.clear(); - if( startsWith( token, "exclude:" ) ) { - m_exclusion = true; - token = token.substr( 8 ); - } - if( !token.empty() ) { - Ptr pattern = new T( token ); - if( m_exclusion ) - pattern = new TestSpec::ExcludedPattern( pattern ); - m_currentFilter.m_patterns.push_back( pattern ); - } - m_exclusion = false; - m_mode = None; - } - void addFilter() { - if( !m_currentFilter.m_patterns.empty() ) { - m_testSpec.m_filters.push_back( m_currentFilter ); - m_currentFilter = TestSpec::Filter(); - } - } - }; - inline TestSpec parseTestSpec( std::string const& arg ) { - return TestSpecParser( ITagAliasRegistry::get() ).parse( arg ).testSpec(); - } - -} // namespace Catch - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - -// #included from: catch_interfaces_config.h -#define TWOBLUECUBES_CATCH_INTERFACES_CONFIG_H_INCLUDED - -#include -#include -#include - -namespace Catch { - - struct Verbosity { enum Level { - NoOutput = 0, - Quiet, - Normal - }; }; - - struct WarnAbout { enum What { - Nothing = 0x00, - NoAssertions = 0x01 - }; }; - - struct ShowDurations { enum OrNot { - DefaultForReporter, - Always, - Never - }; }; - struct RunTests { enum InWhatOrder { - InDeclarationOrder, - InLexicographicalOrder, - InRandomOrder - }; }; - struct UseColour { enum YesOrNo { - Auto, - Yes, - No - }; }; - struct WaitForKeypress { enum When { - Never, - BeforeStart = 1, - BeforeExit = 2, - BeforeStartAndExit = BeforeStart | BeforeExit - }; }; - - class TestSpec; - - struct IConfig : IShared { - - virtual ~IConfig(); - - virtual bool allowThrows() const = 0; - virtual std::ostream& stream() const = 0; - virtual std::string name() const = 0; - virtual bool includeSuccessfulResults() const = 0; - virtual bool shouldDebugBreak() const = 0; - virtual bool warnAboutMissingAssertions() const = 0; - virtual int abortAfter() const = 0; - virtual bool showInvisibles() const = 0; - virtual ShowDurations::OrNot showDurations() const = 0; - virtual TestSpec const& testSpec() const = 0; - virtual RunTests::InWhatOrder runOrder() const = 0; - virtual unsigned int rngSeed() const = 0; - virtual UseColour::YesOrNo useColour() const = 0; - virtual std::vector const& getSectionsToRun() const = 0; - - }; -} - -// #included from: catch_stream.h -#define TWOBLUECUBES_CATCH_STREAM_H_INCLUDED - -// #included from: catch_streambuf.h -#define TWOBLUECUBES_CATCH_STREAMBUF_H_INCLUDED - -#include - -namespace Catch { - - class StreamBufBase : public std::streambuf { - public: - virtual ~StreamBufBase() CATCH_NOEXCEPT; - }; -} - -#include -#include -#include -#include - -namespace Catch { - - std::ostream& cout(); - std::ostream& cerr(); - std::ostream& clog(); - - struct IStream { - virtual ~IStream() CATCH_NOEXCEPT; - virtual std::ostream& stream() const = 0; - }; - - class FileStream : public IStream { - mutable std::ofstream m_ofs; - public: - FileStream( std::string const& filename ); - virtual ~FileStream() CATCH_NOEXCEPT; - public: // IStream - virtual std::ostream& stream() const CATCH_OVERRIDE; - }; - - class CoutStream : public IStream { - mutable std::ostream m_os; - public: - CoutStream(); - virtual ~CoutStream() CATCH_NOEXCEPT; - - public: // IStream - virtual std::ostream& stream() const CATCH_OVERRIDE; - }; - - class DebugOutStream : public IStream { - CATCH_AUTO_PTR( StreamBufBase ) m_streamBuf; - mutable std::ostream m_os; - public: - DebugOutStream(); - virtual ~DebugOutStream() CATCH_NOEXCEPT; - - public: // IStream - virtual std::ostream& stream() const CATCH_OVERRIDE; - }; -} - -#include -#include -#include -#include - -#ifndef CATCH_CONFIG_CONSOLE_WIDTH -#define CATCH_CONFIG_CONSOLE_WIDTH 80 -#endif - -namespace Catch { - - struct ConfigData { - - ConfigData() - : listTests( false ), - listTags( false ), - listReporters( false ), - listTestNamesOnly( false ), - listExtraInfo( false ), - showSuccessfulTests( false ), - shouldDebugBreak( false ), - noThrow( false ), - showHelp( false ), - showInvisibles( false ), - filenamesAsTags( false ), - libIdentify( false ), - abortAfter( -1 ), - rngSeed( 0 ), - verbosity( Verbosity::Normal ), - warnings( WarnAbout::Nothing ), - showDurations( ShowDurations::DefaultForReporter ), - runOrder( RunTests::InDeclarationOrder ), - useColour( UseColour::Auto ), - waitForKeypress( WaitForKeypress::Never ) - {} - - bool listTests; - bool listTags; - bool listReporters; - bool listTestNamesOnly; - bool listExtraInfo; - - bool showSuccessfulTests; - bool shouldDebugBreak; - bool noThrow; - bool showHelp; - bool showInvisibles; - bool filenamesAsTags; - bool libIdentify; - - int abortAfter; - unsigned int rngSeed; - - Verbosity::Level verbosity; - WarnAbout::What warnings; - ShowDurations::OrNot showDurations; - RunTests::InWhatOrder runOrder; - UseColour::YesOrNo useColour; - WaitForKeypress::When waitForKeypress; - - std::string outputFilename; - std::string name; - std::string processName; - - std::vector reporterNames; - std::vector testsOrTags; - std::vector sectionsToRun; - }; - - class Config : public SharedImpl { - private: - Config( Config const& other ); - Config& operator = ( Config const& other ); - virtual void dummy(); - public: - - Config() - {} - - Config( ConfigData const& data ) - : m_data( data ), - m_stream( openStream() ) - { - if( !data.testsOrTags.empty() ) { - TestSpecParser parser( ITagAliasRegistry::get() ); - for( std::size_t i = 0; i < data.testsOrTags.size(); ++i ) - parser.parse( data.testsOrTags[i] ); - m_testSpec = parser.testSpec(); - } - } - - virtual ~Config() {} - - std::string const& getFilename() const { - return m_data.outputFilename ; - } - - bool listTests() const { return m_data.listTests; } - bool listTestNamesOnly() const { return m_data.listTestNamesOnly; } - bool listTags() const { return m_data.listTags; } - bool listReporters() const { return m_data.listReporters; } - bool listExtraInfo() const { return m_data.listExtraInfo; } - - std::string getProcessName() const { return m_data.processName; } - - std::vector const& getReporterNames() const { return m_data.reporterNames; } - std::vector const& getSectionsToRun() const CATCH_OVERRIDE { return m_data.sectionsToRun; } - - virtual TestSpec const& testSpec() const CATCH_OVERRIDE { return m_testSpec; } - - bool showHelp() const { return m_data.showHelp; } - - // IConfig interface - virtual bool allowThrows() const CATCH_OVERRIDE { return !m_data.noThrow; } - virtual std::ostream& stream() const CATCH_OVERRIDE { return m_stream->stream(); } - virtual std::string name() const CATCH_OVERRIDE { return m_data.name.empty() ? m_data.processName : m_data.name; } - virtual bool includeSuccessfulResults() const CATCH_OVERRIDE { return m_data.showSuccessfulTests; } - virtual bool warnAboutMissingAssertions() const CATCH_OVERRIDE { return m_data.warnings & WarnAbout::NoAssertions; } - virtual ShowDurations::OrNot showDurations() const CATCH_OVERRIDE { return m_data.showDurations; } - virtual RunTests::InWhatOrder runOrder() const CATCH_OVERRIDE { return m_data.runOrder; } - virtual unsigned int rngSeed() const CATCH_OVERRIDE { return m_data.rngSeed; } - virtual UseColour::YesOrNo useColour() const CATCH_OVERRIDE { return m_data.useColour; } - virtual bool shouldDebugBreak() const CATCH_OVERRIDE { return m_data.shouldDebugBreak; } - virtual int abortAfter() const CATCH_OVERRIDE { return m_data.abortAfter; } - virtual bool showInvisibles() const CATCH_OVERRIDE { return m_data.showInvisibles; } - - private: - - IStream const* openStream() { - if( m_data.outputFilename.empty() ) - return new CoutStream(); - else if( m_data.outputFilename[0] == '%' ) { - if( m_data.outputFilename == "%debug" ) - return new DebugOutStream(); - else - throw std::domain_error( "Unrecognised stream: " + m_data.outputFilename ); - } - else - return new FileStream( m_data.outputFilename ); - } - ConfigData m_data; - - CATCH_AUTO_PTR( IStream const ) m_stream; - TestSpec m_testSpec; - }; - -} // end namespace Catch - -// #included from: catch_clara.h -#define TWOBLUECUBES_CATCH_CLARA_H_INCLUDED - -// Use Catch's value for console width (store Clara's off to the side, if present) -#ifdef CLARA_CONFIG_CONSOLE_WIDTH -#define CATCH_TEMP_CLARA_CONFIG_CONSOLE_WIDTH CLARA_CONFIG_CONSOLE_WIDTH -#undef CLARA_CONFIG_CONSOLE_WIDTH -#endif -#define CLARA_CONFIG_CONSOLE_WIDTH CATCH_CONFIG_CONSOLE_WIDTH - -// Declare Clara inside the Catch namespace -#define STITCH_CLARA_OPEN_NAMESPACE namespace Catch { -// #included from: ../external/clara.h - -// Version 0.0.2.4 - -// Only use header guard if we are not using an outer namespace -#if !defined(TWOBLUECUBES_CLARA_H_INCLUDED) || defined(STITCH_CLARA_OPEN_NAMESPACE) - -#ifndef STITCH_CLARA_OPEN_NAMESPACE -#define TWOBLUECUBES_CLARA_H_INCLUDED -#define STITCH_CLARA_OPEN_NAMESPACE -#define STITCH_CLARA_CLOSE_NAMESPACE -#else -#define STITCH_CLARA_CLOSE_NAMESPACE } -#endif - -#define STITCH_TBC_TEXT_FORMAT_OPEN_NAMESPACE STITCH_CLARA_OPEN_NAMESPACE - -// ----------- #included from tbc_text_format.h ----------- - -// Only use header guard if we are not using an outer namespace -#if !defined(TBC_TEXT_FORMAT_H_INCLUDED) || defined(STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE) -#ifndef STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE -#define TBC_TEXT_FORMAT_H_INCLUDED -#endif - -#include -#include -#include -#include -#include - -// Use optional outer namespace -#ifdef STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE -namespace STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE { -#endif - -namespace Tbc { - -#ifdef TBC_TEXT_FORMAT_CONSOLE_WIDTH - const unsigned int consoleWidth = TBC_TEXT_FORMAT_CONSOLE_WIDTH; -#else - const unsigned int consoleWidth = 80; -#endif - - struct TextAttributes { - TextAttributes() - : initialIndent( std::string::npos ), - indent( 0 ), - width( consoleWidth-1 ), - tabChar( '\t' ) - {} - - TextAttributes& setInitialIndent( std::size_t _value ) { initialIndent = _value; return *this; } - TextAttributes& setIndent( std::size_t _value ) { indent = _value; return *this; } - TextAttributes& setWidth( std::size_t _value ) { width = _value; return *this; } - TextAttributes& setTabChar( char _value ) { tabChar = _value; return *this; } - - std::size_t initialIndent; // indent of first line, or npos - std::size_t indent; // indent of subsequent lines, or all if initialIndent is npos - std::size_t width; // maximum width of text, including indent. Longer text will wrap - char tabChar; // If this char is seen the indent is changed to current pos - }; - - class Text { - public: - Text( std::string const& _str, TextAttributes const& _attr = TextAttributes() ) - : attr( _attr ) - { - std::string wrappableChars = " [({.,/|\\-"; - std::size_t indent = _attr.initialIndent != std::string::npos - ? _attr.initialIndent - : _attr.indent; - std::string remainder = _str; - - while( !remainder.empty() ) { - if( lines.size() >= 1000 ) { - lines.push_back( "... message truncated due to excessive size" ); - return; - } - std::size_t tabPos = std::string::npos; - std::size_t width = (std::min)( remainder.size(), _attr.width - indent ); - std::size_t pos = remainder.find_first_of( '\n' ); - if( pos <= width ) { - width = pos; - } - pos = remainder.find_last_of( _attr.tabChar, width ); - if( pos != std::string::npos ) { - tabPos = pos; - if( remainder[width] == '\n' ) - width--; - remainder = remainder.substr( 0, tabPos ) + remainder.substr( tabPos+1 ); - } - - if( width == remainder.size() ) { - spliceLine( indent, remainder, width ); - } - else if( remainder[width] == '\n' ) { - spliceLine( indent, remainder, width ); - if( width <= 1 || remainder.size() != 1 ) - remainder = remainder.substr( 1 ); - indent = _attr.indent; - } - else { - pos = remainder.find_last_of( wrappableChars, width ); - if( pos != std::string::npos && pos > 0 ) { - spliceLine( indent, remainder, pos ); - if( remainder[0] == ' ' ) - remainder = remainder.substr( 1 ); - } - else { - spliceLine( indent, remainder, width-1 ); - lines.back() += "-"; - } - if( lines.size() == 1 ) - indent = _attr.indent; - if( tabPos != std::string::npos ) - indent += tabPos; - } - } - } - - void spliceLine( std::size_t _indent, std::string& _remainder, std::size_t _pos ) { - lines.push_back( std::string( _indent, ' ' ) + _remainder.substr( 0, _pos ) ); - _remainder = _remainder.substr( _pos ); - } - - typedef std::vector::const_iterator const_iterator; - - const_iterator begin() const { return lines.begin(); } - const_iterator end() const { return lines.end(); } - std::string const& last() const { return lines.back(); } - std::size_t size() const { return lines.size(); } - std::string const& operator[]( std::size_t _index ) const { return lines[_index]; } - std::string toString() const { - std::ostringstream oss; - oss << *this; - return oss.str(); - } - - friend std::ostream& operator << ( std::ostream& _stream, Text const& _text ) { - for( Text::const_iterator it = _text.begin(), itEnd = _text.end(); - it != itEnd; ++it ) { - if( it != _text.begin() ) - _stream << "\n"; - _stream << *it; - } - return _stream; - } - - private: - std::string str; - TextAttributes attr; - std::vector lines; - }; - -} // end namespace Tbc - -#ifdef STITCH_TBC_TEXT_FORMAT_OUTER_NAMESPACE -} // end outer namespace -#endif - -#endif // TBC_TEXT_FORMAT_H_INCLUDED - -// ----------- end of #include from tbc_text_format.h ----------- -// ........... back in clara.h - -#undef STITCH_TBC_TEXT_FORMAT_OPEN_NAMESPACE - -// ----------- #included from clara_compilers.h ----------- - -#ifndef TWOBLUECUBES_CLARA_COMPILERS_H_INCLUDED -#define TWOBLUECUBES_CLARA_COMPILERS_H_INCLUDED - -// Detect a number of compiler features - mostly C++11/14 conformance - by compiler -// The following features are defined: -// -// CLARA_CONFIG_CPP11_NULLPTR : is nullptr supported? -// CLARA_CONFIG_CPP11_NOEXCEPT : is noexcept supported? -// CLARA_CONFIG_CPP11_GENERATED_METHODS : The delete and default keywords for compiler generated methods -// CLARA_CONFIG_CPP11_OVERRIDE : is override supported? -// CLARA_CONFIG_CPP11_UNIQUE_PTR : is unique_ptr supported (otherwise use auto_ptr) - -// CLARA_CONFIG_CPP11_OR_GREATER : Is C++11 supported? - -// CLARA_CONFIG_VARIADIC_MACROS : are variadic macros supported? - -// In general each macro has a _NO_ form -// (e.g. CLARA_CONFIG_CPP11_NO_NULLPTR) which disables the feature. -// Many features, at point of detection, define an _INTERNAL_ macro, so they -// can be combined, en-mass, with the _NO_ forms later. - -// All the C++11 features can be disabled with CLARA_CONFIG_NO_CPP11 - -#ifdef __clang__ - -#if __has_feature(cxx_nullptr) -#define CLARA_INTERNAL_CONFIG_CPP11_NULLPTR -#endif - -#if __has_feature(cxx_noexcept) -#define CLARA_INTERNAL_CONFIG_CPP11_NOEXCEPT -#endif - -#endif // __clang__ - -//////////////////////////////////////////////////////////////////////////////// -// GCC -#ifdef __GNUC__ - -#if __GNUC__ == 4 && __GNUC_MINOR__ >= 6 && defined(__GXX_EXPERIMENTAL_CXX0X__) -#define CLARA_INTERNAL_CONFIG_CPP11_NULLPTR -#endif - -// - otherwise more recent versions define __cplusplus >= 201103L -// and will get picked up below - -#endif // __GNUC__ - -//////////////////////////////////////////////////////////////////////////////// -// Visual C++ -#ifdef _MSC_VER - -#if (_MSC_VER >= 1600) -#define CLARA_INTERNAL_CONFIG_CPP11_NULLPTR -#define CLARA_INTERNAL_CONFIG_CPP11_UNIQUE_PTR -#endif - -#if (_MSC_VER >= 1900 ) // (VC++ 13 (VS2015)) -#define CLARA_INTERNAL_CONFIG_CPP11_NOEXCEPT -#define CLARA_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -#endif - -#endif // _MSC_VER - -//////////////////////////////////////////////////////////////////////////////// -// C++ language feature support - -// catch all support for C++11 -#if defined(__cplusplus) && __cplusplus >= 201103L - -#define CLARA_CPP11_OR_GREATER - -#if !defined(CLARA_INTERNAL_CONFIG_CPP11_NULLPTR) -#define CLARA_INTERNAL_CONFIG_CPP11_NULLPTR -#endif - -#ifndef CLARA_INTERNAL_CONFIG_CPP11_NOEXCEPT -#define CLARA_INTERNAL_CONFIG_CPP11_NOEXCEPT -#endif - -#ifndef CLARA_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -#define CLARA_INTERNAL_CONFIG_CPP11_GENERATED_METHODS -#endif - -#if !defined(CLARA_INTERNAL_CONFIG_CPP11_OVERRIDE) -#define CLARA_INTERNAL_CONFIG_CPP11_OVERRIDE -#endif -#if !defined(CLARA_INTERNAL_CONFIG_CPP11_UNIQUE_PTR) -#define CLARA_INTERNAL_CONFIG_CPP11_UNIQUE_PTR -#endif - -#endif // __cplusplus >= 201103L - -// Now set the actual defines based on the above + anything the user has configured -#if defined(CLARA_INTERNAL_CONFIG_CPP11_NULLPTR) && !defined(CLARA_CONFIG_CPP11_NO_NULLPTR) && !defined(CLARA_CONFIG_CPP11_NULLPTR) && !defined(CLARA_CONFIG_NO_CPP11) -#define CLARA_CONFIG_CPP11_NULLPTR -#endif -#if defined(CLARA_INTERNAL_CONFIG_CPP11_NOEXCEPT) && !defined(CLARA_CONFIG_CPP11_NO_NOEXCEPT) && !defined(CLARA_CONFIG_CPP11_NOEXCEPT) && !defined(CLARA_CONFIG_NO_CPP11) -#define CLARA_CONFIG_CPP11_NOEXCEPT -#endif -#if defined(CLARA_INTERNAL_CONFIG_CPP11_GENERATED_METHODS) && !defined(CLARA_CONFIG_CPP11_NO_GENERATED_METHODS) && !defined(CLARA_CONFIG_CPP11_GENERATED_METHODS) && !defined(CLARA_CONFIG_NO_CPP11) -#define CLARA_CONFIG_CPP11_GENERATED_METHODS -#endif -#if defined(CLARA_INTERNAL_CONFIG_CPP11_OVERRIDE) && !defined(CLARA_CONFIG_NO_OVERRIDE) && !defined(CLARA_CONFIG_CPP11_OVERRIDE) && !defined(CLARA_CONFIG_NO_CPP11) -#define CLARA_CONFIG_CPP11_OVERRIDE -#endif -#if defined(CLARA_INTERNAL_CONFIG_CPP11_UNIQUE_PTR) && !defined(CLARA_CONFIG_NO_UNIQUE_PTR) && !defined(CLARA_CONFIG_CPP11_UNIQUE_PTR) && !defined(CLARA_CONFIG_NO_CPP11) -#define CLARA_CONFIG_CPP11_UNIQUE_PTR -#endif - -// noexcept support: -#if defined(CLARA_CONFIG_CPP11_NOEXCEPT) && !defined(CLARA_NOEXCEPT) -#define CLARA_NOEXCEPT noexcept -# define CLARA_NOEXCEPT_IS(x) noexcept(x) -#else -#define CLARA_NOEXCEPT throw() -# define CLARA_NOEXCEPT_IS(x) -#endif - -// nullptr support -#ifdef CLARA_CONFIG_CPP11_NULLPTR -#define CLARA_NULL nullptr -#else -#define CLARA_NULL NULL -#endif - -// override support -#ifdef CLARA_CONFIG_CPP11_OVERRIDE -#define CLARA_OVERRIDE override -#else -#define CLARA_OVERRIDE -#endif - -// unique_ptr support -#ifdef CLARA_CONFIG_CPP11_UNIQUE_PTR -# define CLARA_AUTO_PTR( T ) std::unique_ptr -#else -# define CLARA_AUTO_PTR( T ) std::auto_ptr -#endif - -#endif // TWOBLUECUBES_CLARA_COMPILERS_H_INCLUDED - -// ----------- end of #include from clara_compilers.h ----------- -// ........... back in clara.h - -#include -#include -#include - -#if defined(WIN32) || defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER) -#define CLARA_PLATFORM_WINDOWS -#endif - -// Use optional outer namespace -#ifdef STITCH_CLARA_OPEN_NAMESPACE -STITCH_CLARA_OPEN_NAMESPACE -#endif - -namespace Clara { - - struct UnpositionalTag {}; - - extern UnpositionalTag _; - -#ifdef CLARA_CONFIG_MAIN - UnpositionalTag _; -#endif - - namespace Detail { - -#ifdef CLARA_CONSOLE_WIDTH - const unsigned int consoleWidth = CLARA_CONFIG_CONSOLE_WIDTH; -#else - const unsigned int consoleWidth = 80; -#endif - - using namespace Tbc; - - inline bool startsWith( std::string const& str, std::string const& prefix ) { - return str.size() >= prefix.size() && str.substr( 0, prefix.size() ) == prefix; - } - - template struct RemoveConstRef{ typedef T type; }; - template struct RemoveConstRef{ typedef T type; }; - template struct RemoveConstRef{ typedef T type; }; - template struct RemoveConstRef{ typedef T type; }; - - template struct IsBool { static const bool value = false; }; - template<> struct IsBool { static const bool value = true; }; - - template - void convertInto( std::string const& _source, T& _dest ) { - std::stringstream ss; - ss << _source; - ss >> _dest; - if( ss.fail() ) - throw std::runtime_error( "Unable to convert " + _source + " to destination type" ); - } - inline void convertInto( std::string const& _source, std::string& _dest ) { - _dest = _source; - } - char toLowerCh(char c) { - return static_cast( std::tolower( c ) ); - } - inline void convertInto( std::string const& _source, bool& _dest ) { - std::string sourceLC = _source; - std::transform( sourceLC.begin(), sourceLC.end(), sourceLC.begin(), toLowerCh ); - if( sourceLC == "y" || sourceLC == "1" || sourceLC == "true" || sourceLC == "yes" || sourceLC == "on" ) - _dest = true; - else if( sourceLC == "n" || sourceLC == "0" || sourceLC == "false" || sourceLC == "no" || sourceLC == "off" ) - _dest = false; - else - throw std::runtime_error( "Expected a boolean value but did not recognise:\n '" + _source + "'" ); - } - - template - struct IArgFunction { - virtual ~IArgFunction() {} -#ifdef CLARA_CONFIG_CPP11_GENERATED_METHODS - IArgFunction() = default; - IArgFunction( IArgFunction const& ) = default; -#endif - virtual void set( ConfigT& config, std::string const& value ) const = 0; - virtual bool takesArg() const = 0; - virtual IArgFunction* clone() const = 0; - }; - - template - class BoundArgFunction { - public: - BoundArgFunction() : functionObj( CLARA_NULL ) {} - BoundArgFunction( IArgFunction* _functionObj ) : functionObj( _functionObj ) {} - BoundArgFunction( BoundArgFunction const& other ) : functionObj( other.functionObj ? other.functionObj->clone() : CLARA_NULL ) {} - BoundArgFunction& operator = ( BoundArgFunction const& other ) { - IArgFunction* newFunctionObj = other.functionObj ? other.functionObj->clone() : CLARA_NULL; - delete functionObj; - functionObj = newFunctionObj; - return *this; - } - ~BoundArgFunction() { delete functionObj; } - - void set( ConfigT& config, std::string const& value ) const { - functionObj->set( config, value ); - } - bool takesArg() const { return functionObj->takesArg(); } - - bool isSet() const { - return functionObj != CLARA_NULL; - } - private: - IArgFunction* functionObj; - }; - - template - struct NullBinder : IArgFunction{ - virtual void set( C&, std::string const& ) const {} - virtual bool takesArg() const { return true; } - virtual IArgFunction* clone() const { return new NullBinder( *this ); } - }; - - template - struct BoundDataMember : IArgFunction{ - BoundDataMember( M C::* _member ) : member( _member ) {} - virtual void set( C& p, std::string const& stringValue ) const { - convertInto( stringValue, p.*member ); - } - virtual bool takesArg() const { return !IsBool::value; } - virtual IArgFunction* clone() const { return new BoundDataMember( *this ); } - M C::* member; - }; - template - struct BoundUnaryMethod : IArgFunction{ - BoundUnaryMethod( void (C::*_member)( M ) ) : member( _member ) {} - virtual void set( C& p, std::string const& stringValue ) const { - typename RemoveConstRef::type value; - convertInto( stringValue, value ); - (p.*member)( value ); - } - virtual bool takesArg() const { return !IsBool::value; } - virtual IArgFunction* clone() const { return new BoundUnaryMethod( *this ); } - void (C::*member)( M ); - }; - template - struct BoundNullaryMethod : IArgFunction{ - BoundNullaryMethod( void (C::*_member)() ) : member( _member ) {} - virtual void set( C& p, std::string const& stringValue ) const { - bool value; - convertInto( stringValue, value ); - if( value ) - (p.*member)(); - } - virtual bool takesArg() const { return false; } - virtual IArgFunction* clone() const { return new BoundNullaryMethod( *this ); } - void (C::*member)(); - }; - - template - struct BoundUnaryFunction : IArgFunction{ - BoundUnaryFunction( void (*_function)( C& ) ) : function( _function ) {} - virtual void set( C& obj, std::string const& stringValue ) const { - bool value; - convertInto( stringValue, value ); - if( value ) - function( obj ); - } - virtual bool takesArg() const { return false; } - virtual IArgFunction* clone() const { return new BoundUnaryFunction( *this ); } - void (*function)( C& ); - }; - - template - struct BoundBinaryFunction : IArgFunction{ - BoundBinaryFunction( void (*_function)( C&, T ) ) : function( _function ) {} - virtual void set( C& obj, std::string const& stringValue ) const { - typename RemoveConstRef::type value; - convertInto( stringValue, value ); - function( obj, value ); - } - virtual bool takesArg() const { return !IsBool::value; } - virtual IArgFunction* clone() const { return new BoundBinaryFunction( *this ); } - void (*function)( C&, T ); - }; - - } // namespace Detail - - inline std::vector argsToVector( int argc, char const* const* const argv ) { - std::vector args( static_cast( argc ) ); - for( std::size_t i = 0; i < static_cast( argc ); ++i ) - args[i] = argv[i]; - - return args; - } - - class Parser { - enum Mode { None, MaybeShortOpt, SlashOpt, ShortOpt, LongOpt, Positional }; - Mode mode; - std::size_t from; - bool inQuotes; - public: - - struct Token { - enum Type { Positional, ShortOpt, LongOpt }; - Token( Type _type, std::string const& _data ) : type( _type ), data( _data ) {} - Type type; - std::string data; - }; - - Parser() : mode( None ), from( 0 ), inQuotes( false ){} - - void parseIntoTokens( std::vector const& args, std::vector& tokens ) { - const std::string doubleDash = "--"; - for( std::size_t i = 1; i < args.size() && args[i] != doubleDash; ++i ) - parseIntoTokens( args[i], tokens); - } - - void parseIntoTokens( std::string const& arg, std::vector& tokens ) { - for( std::size_t i = 0; i < arg.size(); ++i ) { - char c = arg[i]; - if( c == '"' ) - inQuotes = !inQuotes; - mode = handleMode( i, c, arg, tokens ); - } - mode = handleMode( arg.size(), '\0', arg, tokens ); - } - Mode handleMode( std::size_t i, char c, std::string const& arg, std::vector& tokens ) { - switch( mode ) { - case None: return handleNone( i, c ); - case MaybeShortOpt: return handleMaybeShortOpt( i, c ); - case ShortOpt: - case LongOpt: - case SlashOpt: return handleOpt( i, c, arg, tokens ); - case Positional: return handlePositional( i, c, arg, tokens ); - default: throw std::logic_error( "Unknown mode" ); - } - } - - Mode handleNone( std::size_t i, char c ) { - if( inQuotes ) { - from = i; - return Positional; - } - switch( c ) { - case '-': return MaybeShortOpt; -#ifdef CLARA_PLATFORM_WINDOWS - case '/': from = i+1; return SlashOpt; -#endif - default: from = i; return Positional; - } - } - Mode handleMaybeShortOpt( std::size_t i, char c ) { - switch( c ) { - case '-': from = i+1; return LongOpt; - default: from = i; return ShortOpt; - } - } - - Mode handleOpt( std::size_t i, char c, std::string const& arg, std::vector& tokens ) { - if( std::string( ":=\0", 3 ).find( c ) == std::string::npos ) - return mode; - - std::string optName = arg.substr( from, i-from ); - if( mode == ShortOpt ) - for( std::size_t j = 0; j < optName.size(); ++j ) - tokens.push_back( Token( Token::ShortOpt, optName.substr( j, 1 ) ) ); - else if( mode == SlashOpt && optName.size() == 1 ) - tokens.push_back( Token( Token::ShortOpt, optName ) ); - else - tokens.push_back( Token( Token::LongOpt, optName ) ); - return None; - } - Mode handlePositional( std::size_t i, char c, std::string const& arg, std::vector& tokens ) { - if( inQuotes || std::string( "\0", 1 ).find( c ) == std::string::npos ) - return mode; - - std::string data = arg.substr( from, i-from ); - tokens.push_back( Token( Token::Positional, data ) ); - return None; - } - }; - - template - struct CommonArgProperties { - CommonArgProperties() {} - CommonArgProperties( Detail::BoundArgFunction const& _boundField ) : boundField( _boundField ) {} - - Detail::BoundArgFunction boundField; - std::string description; - std::string detail; - std::string placeholder; // Only value if boundField takes an arg - - bool takesArg() const { - return !placeholder.empty(); - } - void validate() const { - if( !boundField.isSet() ) - throw std::logic_error( "option not bound" ); - } - }; - struct OptionArgProperties { - std::vector shortNames; - std::string longName; - - bool hasShortName( std::string const& shortName ) const { - return std::find( shortNames.begin(), shortNames.end(), shortName ) != shortNames.end(); - } - bool hasLongName( std::string const& _longName ) const { - return _longName == longName; - } - }; - struct PositionalArgProperties { - PositionalArgProperties() : position( -1 ) {} - int position; // -1 means non-positional (floating) - - bool isFixedPositional() const { - return position != -1; - } - }; - - template - class CommandLine { - - struct Arg : CommonArgProperties, OptionArgProperties, PositionalArgProperties { - Arg() {} - Arg( Detail::BoundArgFunction const& _boundField ) : CommonArgProperties( _boundField ) {} - - using CommonArgProperties::placeholder; // !TBD - - std::string dbgName() const { - if( !longName.empty() ) - return "--" + longName; - if( !shortNames.empty() ) - return "-" + shortNames[0]; - return "positional args"; - } - std::string commands() const { - std::ostringstream oss; - bool first = true; - std::vector::const_iterator it = shortNames.begin(), itEnd = shortNames.end(); - for(; it != itEnd; ++it ) { - if( first ) - first = false; - else - oss << ", "; - oss << "-" << *it; - } - if( !longName.empty() ) { - if( !first ) - oss << ", "; - oss << "--" << longName; - } - if( !placeholder.empty() ) - oss << " <" << placeholder << ">"; - return oss.str(); - } - }; - - typedef CLARA_AUTO_PTR( Arg ) ArgAutoPtr; - - friend void addOptName( Arg& arg, std::string const& optName ) - { - if( optName.empty() ) - return; - if( Detail::startsWith( optName, "--" ) ) { - if( !arg.longName.empty() ) - throw std::logic_error( "Only one long opt may be specified. '" - + arg.longName - + "' already specified, now attempting to add '" - + optName + "'" ); - arg.longName = optName.substr( 2 ); - } - else if( Detail::startsWith( optName, "-" ) ) - arg.shortNames.push_back( optName.substr( 1 ) ); - else - throw std::logic_error( "option must begin with - or --. Option was: '" + optName + "'" ); - } - friend void setPositionalArg( Arg& arg, int position ) - { - arg.position = position; - } - - class ArgBuilder { - public: - ArgBuilder( Arg* arg ) : m_arg( arg ) {} - - // Bind a non-boolean data member (requires placeholder string) - template - void bind( M C::* field, std::string const& placeholder ) { - m_arg->boundField = new Detail::BoundDataMember( field ); - m_arg->placeholder = placeholder; - } - // Bind a boolean data member (no placeholder required) - template - void bind( bool C::* field ) { - m_arg->boundField = new Detail::BoundDataMember( field ); - } - - // Bind a method taking a single, non-boolean argument (requires a placeholder string) - template - void bind( void (C::* unaryMethod)( M ), std::string const& placeholder ) { - m_arg->boundField = new Detail::BoundUnaryMethod( unaryMethod ); - m_arg->placeholder = placeholder; - } - - // Bind a method taking a single, boolean argument (no placeholder string required) - template - void bind( void (C::* unaryMethod)( bool ) ) { - m_arg->boundField = new Detail::BoundUnaryMethod( unaryMethod ); - } - - // Bind a method that takes no arguments (will be called if opt is present) - template - void bind( void (C::* nullaryMethod)() ) { - m_arg->boundField = new Detail::BoundNullaryMethod( nullaryMethod ); - } - - // Bind a free function taking a single argument - the object to operate on (no placeholder string required) - template - void bind( void (* unaryFunction)( C& ) ) { - m_arg->boundField = new Detail::BoundUnaryFunction( unaryFunction ); - } - - // Bind a free function taking a single argument - the object to operate on (requires a placeholder string) - template - void bind( void (* binaryFunction)( C&, T ), std::string const& placeholder ) { - m_arg->boundField = new Detail::BoundBinaryFunction( binaryFunction ); - m_arg->placeholder = placeholder; - } - - ArgBuilder& describe( std::string const& description ) { - m_arg->description = description; - return *this; - } - ArgBuilder& detail( std::string const& detail ) { - m_arg->detail = detail; - return *this; - } - - protected: - Arg* m_arg; - }; - - class OptBuilder : public ArgBuilder { - public: - OptBuilder( Arg* arg ) : ArgBuilder( arg ) {} - OptBuilder( OptBuilder& other ) : ArgBuilder( other ) {} - - OptBuilder& operator[]( std::string const& optName ) { - addOptName( *ArgBuilder::m_arg, optName ); - return *this; - } - }; - - public: - - CommandLine() - : m_boundProcessName( new Detail::NullBinder() ), - m_highestSpecifiedArgPosition( 0 ), - m_throwOnUnrecognisedTokens( false ) - {} - CommandLine( CommandLine const& other ) - : m_boundProcessName( other.m_boundProcessName ), - m_options ( other.m_options ), - m_positionalArgs( other.m_positionalArgs ), - m_highestSpecifiedArgPosition( other.m_highestSpecifiedArgPosition ), - m_throwOnUnrecognisedTokens( other.m_throwOnUnrecognisedTokens ) - { - if( other.m_floatingArg.get() ) - m_floatingArg.reset( new Arg( *other.m_floatingArg ) ); - } - - CommandLine& setThrowOnUnrecognisedTokens( bool shouldThrow = true ) { - m_throwOnUnrecognisedTokens = shouldThrow; - return *this; - } - - OptBuilder operator[]( std::string const& optName ) { - m_options.push_back( Arg() ); - addOptName( m_options.back(), optName ); - OptBuilder builder( &m_options.back() ); - return builder; - } - - ArgBuilder operator[]( int position ) { - m_positionalArgs.insert( std::make_pair( position, Arg() ) ); - if( position > m_highestSpecifiedArgPosition ) - m_highestSpecifiedArgPosition = position; - setPositionalArg( m_positionalArgs[position], position ); - ArgBuilder builder( &m_positionalArgs[position] ); - return builder; - } - - // Invoke this with the _ instance - ArgBuilder operator[]( UnpositionalTag ) { - if( m_floatingArg.get() ) - throw std::logic_error( "Only one unpositional argument can be added" ); - m_floatingArg.reset( new Arg() ); - ArgBuilder builder( m_floatingArg.get() ); - return builder; - } - - template - void bindProcessName( M C::* field ) { - m_boundProcessName = new Detail::BoundDataMember( field ); - } - template - void bindProcessName( void (C::*_unaryMethod)( M ) ) { - m_boundProcessName = new Detail::BoundUnaryMethod( _unaryMethod ); - } - - void optUsage( std::ostream& os, std::size_t indent = 0, std::size_t width = Detail::consoleWidth ) const { - typename std::vector::const_iterator itBegin = m_options.begin(), itEnd = m_options.end(), it; - std::size_t maxWidth = 0; - for( it = itBegin; it != itEnd; ++it ) - maxWidth = (std::max)( maxWidth, it->commands().size() ); - - for( it = itBegin; it != itEnd; ++it ) { - Detail::Text usage( it->commands(), Detail::TextAttributes() - .setWidth( maxWidth+indent ) - .setIndent( indent ) ); - Detail::Text desc( it->description, Detail::TextAttributes() - .setWidth( width - maxWidth - 3 ) ); - - for( std::size_t i = 0; i < (std::max)( usage.size(), desc.size() ); ++i ) { - std::string usageCol = i < usage.size() ? usage[i] : ""; - os << usageCol; - - if( i < desc.size() && !desc[i].empty() ) - os << std::string( indent + 2 + maxWidth - usageCol.size(), ' ' ) - << desc[i]; - os << "\n"; - } - } - } - std::string optUsage() const { - std::ostringstream oss; - optUsage( oss ); - return oss.str(); - } - - void argSynopsis( std::ostream& os ) const { - for( int i = 1; i <= m_highestSpecifiedArgPosition; ++i ) { - if( i > 1 ) - os << " "; - typename std::map::const_iterator it = m_positionalArgs.find( i ); - if( it != m_positionalArgs.end() ) - os << "<" << it->second.placeholder << ">"; - else if( m_floatingArg.get() ) - os << "<" << m_floatingArg->placeholder << ">"; - else - throw std::logic_error( "non consecutive positional arguments with no floating args" ); - } - // !TBD No indication of mandatory args - if( m_floatingArg.get() ) { - if( m_highestSpecifiedArgPosition > 1 ) - os << " "; - os << "[<" << m_floatingArg->placeholder << "> ...]"; - } - } - std::string argSynopsis() const { - std::ostringstream oss; - argSynopsis( oss ); - return oss.str(); - } - - void usage( std::ostream& os, std::string const& procName ) const { - validate(); - os << "usage:\n " << procName << " "; - argSynopsis( os ); - if( !m_options.empty() ) { - os << " [options]\n\nwhere options are: \n"; - optUsage( os, 2 ); - } - os << "\n"; - } - std::string usage( std::string const& procName ) const { - std::ostringstream oss; - usage( oss, procName ); - return oss.str(); - } - - ConfigT parse( std::vector const& args ) const { - ConfigT config; - parseInto( args, config ); - return config; - } - - std::vector parseInto( std::vector const& args, ConfigT& config ) const { - std::string processName = args.empty() ? std::string() : args[0]; - std::size_t lastSlash = processName.find_last_of( "/\\" ); - if( lastSlash != std::string::npos ) - processName = processName.substr( lastSlash+1 ); - m_boundProcessName.set( config, processName ); - std::vector tokens; - Parser parser; - parser.parseIntoTokens( args, tokens ); - return populate( tokens, config ); - } - - std::vector populate( std::vector const& tokens, ConfigT& config ) const { - validate(); - std::vector unusedTokens = populateOptions( tokens, config ); - unusedTokens = populateFixedArgs( unusedTokens, config ); - unusedTokens = populateFloatingArgs( unusedTokens, config ); - return unusedTokens; - } - - std::vector populateOptions( std::vector const& tokens, ConfigT& config ) const { - std::vector unusedTokens; - std::vector errors; - for( std::size_t i = 0; i < tokens.size(); ++i ) { - Parser::Token const& token = tokens[i]; - typename std::vector::const_iterator it = m_options.begin(), itEnd = m_options.end(); - for(; it != itEnd; ++it ) { - Arg const& arg = *it; - - try { - if( ( token.type == Parser::Token::ShortOpt && arg.hasShortName( token.data ) ) || - ( token.type == Parser::Token::LongOpt && arg.hasLongName( token.data ) ) ) { - if( arg.takesArg() ) { - if( i == tokens.size()-1 || tokens[i+1].type != Parser::Token::Positional ) - errors.push_back( "Expected argument to option: " + token.data ); - else - arg.boundField.set( config, tokens[++i].data ); - } - else { - arg.boundField.set( config, "true" ); - } - break; - } - } - catch( std::exception& ex ) { - errors.push_back( std::string( ex.what() ) + "\n- while parsing: (" + arg.commands() + ")" ); - } - } - if( it == itEnd ) { - if( token.type == Parser::Token::Positional || !m_throwOnUnrecognisedTokens ) - unusedTokens.push_back( token ); - else if( errors.empty() && m_throwOnUnrecognisedTokens ) - errors.push_back( "unrecognised option: " + token.data ); - } - } - if( !errors.empty() ) { - std::ostringstream oss; - for( std::vector::const_iterator it = errors.begin(), itEnd = errors.end(); - it != itEnd; - ++it ) { - if( it != errors.begin() ) - oss << "\n"; - oss << *it; - } - throw std::runtime_error( oss.str() ); - } - return unusedTokens; - } - std::vector populateFixedArgs( std::vector const& tokens, ConfigT& config ) const { - std::vector unusedTokens; - int position = 1; - for( std::size_t i = 0; i < tokens.size(); ++i ) { - Parser::Token const& token = tokens[i]; - typename std::map::const_iterator it = m_positionalArgs.find( position ); - if( it != m_positionalArgs.end() ) - it->second.boundField.set( config, token.data ); - else - unusedTokens.push_back( token ); - if( token.type == Parser::Token::Positional ) - position++; - } - return unusedTokens; - } - std::vector populateFloatingArgs( std::vector const& tokens, ConfigT& config ) const { - if( !m_floatingArg.get() ) - return tokens; - std::vector unusedTokens; - for( std::size_t i = 0; i < tokens.size(); ++i ) { - Parser::Token const& token = tokens[i]; - if( token.type == Parser::Token::Positional ) - m_floatingArg->boundField.set( config, token.data ); - else - unusedTokens.push_back( token ); - } - return unusedTokens; - } - - void validate() const - { - if( m_options.empty() && m_positionalArgs.empty() && !m_floatingArg.get() ) - throw std::logic_error( "No options or arguments specified" ); - - for( typename std::vector::const_iterator it = m_options.begin(), - itEnd = m_options.end(); - it != itEnd; ++it ) - it->validate(); - } - - private: - Detail::BoundArgFunction m_boundProcessName; - std::vector m_options; - std::map m_positionalArgs; - ArgAutoPtr m_floatingArg; - int m_highestSpecifiedArgPosition; - bool m_throwOnUnrecognisedTokens; - }; - -} // end namespace Clara - -STITCH_CLARA_CLOSE_NAMESPACE -#undef STITCH_CLARA_OPEN_NAMESPACE -#undef STITCH_CLARA_CLOSE_NAMESPACE - -#endif // TWOBLUECUBES_CLARA_H_INCLUDED -#undef STITCH_CLARA_OPEN_NAMESPACE - -// Restore Clara's value for console width, if present -#ifdef CATCH_TEMP_CLARA_CONFIG_CONSOLE_WIDTH -#define CLARA_CONFIG_CONSOLE_WIDTH CATCH_TEMP_CLARA_CONFIG_CONSOLE_WIDTH -#undef CATCH_TEMP_CLARA_CONFIG_CONSOLE_WIDTH -#endif - -#include -#include - -namespace Catch { - - inline void abortAfterFirst( ConfigData& config ) { config.abortAfter = 1; } - inline void abortAfterX( ConfigData& config, int x ) { - if( x < 1 ) - throw std::runtime_error( "Value after -x or --abortAfter must be greater than zero" ); - config.abortAfter = x; - } - inline void addTestOrTags( ConfigData& config, std::string const& _testSpec ) { config.testsOrTags.push_back( _testSpec ); } - inline void addSectionToRun( ConfigData& config, std::string const& sectionName ) { config.sectionsToRun.push_back( sectionName ); } - inline void addReporterName( ConfigData& config, std::string const& _reporterName ) { config.reporterNames.push_back( _reporterName ); } - - inline void addWarning( ConfigData& config, std::string const& _warning ) { - if( _warning == "NoAssertions" ) - config.warnings = static_cast( config.warnings | WarnAbout::NoAssertions ); - else - throw std::runtime_error( "Unrecognised warning: '" + _warning + '\'' ); - } - inline void setOrder( ConfigData& config, std::string const& order ) { - if( startsWith( "declared", order ) ) - config.runOrder = RunTests::InDeclarationOrder; - else if( startsWith( "lexical", order ) ) - config.runOrder = RunTests::InLexicographicalOrder; - else if( startsWith( "random", order ) ) - config.runOrder = RunTests::InRandomOrder; - else - throw std::runtime_error( "Unrecognised ordering: '" + order + '\'' ); - } - inline void setRngSeed( ConfigData& config, std::string const& seed ) { - if( seed == "time" ) { - config.rngSeed = static_cast( std::time(0) ); - } - else { - std::stringstream ss; - ss << seed; - ss >> config.rngSeed; - if( ss.fail() ) - throw std::runtime_error( "Argument to --rng-seed should be the word 'time' or a number" ); - } - } - inline void setVerbosity( ConfigData& config, int level ) { - // !TBD: accept strings? - config.verbosity = static_cast( level ); - } - inline void setShowDurations( ConfigData& config, bool _showDurations ) { - config.showDurations = _showDurations - ? ShowDurations::Always - : ShowDurations::Never; - } - inline void setUseColour( ConfigData& config, std::string const& value ) { - std::string mode = toLower( value ); - - if( mode == "yes" ) - config.useColour = UseColour::Yes; - else if( mode == "no" ) - config.useColour = UseColour::No; - else if( mode == "auto" ) - config.useColour = UseColour::Auto; - else - throw std::runtime_error( "colour mode must be one of: auto, yes or no" ); - } - inline void setWaitForKeypress( ConfigData& config, std::string const& keypress ) { - std::string keypressLc = toLower( keypress ); - if( keypressLc == "start" ) - config.waitForKeypress = WaitForKeypress::BeforeStart; - else if( keypressLc == "exit" ) - config.waitForKeypress = WaitForKeypress::BeforeExit; - else if( keypressLc == "both" ) - config.waitForKeypress = WaitForKeypress::BeforeStartAndExit; - else - throw std::runtime_error( "keypress argument must be one of: start, exit or both. '" + keypress + "' not recognised" ); - }; - - inline void forceColour( ConfigData& config ) { - config.useColour = UseColour::Yes; - } - inline void loadTestNamesFromFile( ConfigData& config, std::string const& _filename ) { - std::ifstream f( _filename.c_str() ); - if( !f.is_open() ) - throw std::domain_error( "Unable to load input file: " + _filename ); - - std::string line; - while( std::getline( f, line ) ) { - line = trim(line); - if( !line.empty() && !startsWith( line, '#' ) ) { - if( !startsWith( line, '"' ) ) - line = '"' + line + '"'; - addTestOrTags( config, line + ',' ); - } - } - } - - inline Clara::CommandLine makeCommandLineParser() { - - using namespace Clara; - CommandLine cli; - - cli.bindProcessName( &ConfigData::processName ); - - cli["-?"]["-h"]["--help"] - .describe( "display usage information" ) - .bind( &ConfigData::showHelp ); - - cli["-l"]["--list-tests"] - .describe( "list all/matching test cases" ) - .bind( &ConfigData::listTests ); - - cli["-t"]["--list-tags"] - .describe( "list all/matching tags" ) - .bind( &ConfigData::listTags ); - - cli["-s"]["--success"] - .describe( "include successful tests in output" ) - .bind( &ConfigData::showSuccessfulTests ); - - cli["-b"]["--break"] - .describe( "break into debugger on failure" ) - .bind( &ConfigData::shouldDebugBreak ); - - cli["-e"]["--nothrow"] - .describe( "skip exception tests" ) - .bind( &ConfigData::noThrow ); - - cli["-i"]["--invisibles"] - .describe( "show invisibles (tabs, newlines)" ) - .bind( &ConfigData::showInvisibles ); - - cli["-o"]["--out"] - .describe( "output filename" ) - .bind( &ConfigData::outputFilename, "filename" ); - - cli["-r"]["--reporter"] -// .placeholder( "name[:filename]" ) - .describe( "reporter to use (defaults to console)" ) - .bind( &addReporterName, "name" ); - - cli["-n"]["--name"] - .describe( "suite name" ) - .bind( &ConfigData::name, "name" ); - - cli["-a"]["--abort"] - .describe( "abort at first failure" ) - .bind( &abortAfterFirst ); - - cli["-x"]["--abortx"] - .describe( "abort after x failures" ) - .bind( &abortAfterX, "no. failures" ); - - cli["-w"]["--warn"] - .describe( "enable warnings" ) - .bind( &addWarning, "warning name" ); - -// - needs updating if reinstated -// cli.into( &setVerbosity ) -// .describe( "level of verbosity (0=no output)" ) -// .shortOpt( "v") -// .longOpt( "verbosity" ) -// .placeholder( "level" ); - - cli[_] - .describe( "which test or tests to use" ) - .bind( &addTestOrTags, "test name, pattern or tags" ); - - cli["-d"]["--durations"] - .describe( "show test durations" ) - .bind( &setShowDurations, "yes|no" ); - - cli["-f"]["--input-file"] - .describe( "load test names to run from a file" ) - .bind( &loadTestNamesFromFile, "filename" ); - - cli["-#"]["--filenames-as-tags"] - .describe( "adds a tag for the filename" ) - .bind( &ConfigData::filenamesAsTags ); - - cli["-c"]["--section"] - .describe( "specify section to run" ) - .bind( &addSectionToRun, "section name" ); - - // Less common commands which don't have a short form - cli["--list-test-names-only"] - .describe( "list all/matching test cases names only" ) - .bind( &ConfigData::listTestNamesOnly ); - - cli["--list-extra-info"] - .describe( "list all/matching test cases with more info" ) - .bind( &ConfigData::listExtraInfo ); - - cli["--list-reporters"] - .describe( "list all reporters" ) - .bind( &ConfigData::listReporters ); - - cli["--order"] - .describe( "test case order (defaults to decl)" ) - .bind( &setOrder, "decl|lex|rand" ); - - cli["--rng-seed"] - .describe( "set a specific seed for random numbers" ) - .bind( &setRngSeed, "'time'|number" ); - - cli["--force-colour"] - .describe( "force colourised output (deprecated)" ) - .bind( &forceColour ); - - cli["--use-colour"] - .describe( "should output be colourised" ) - .bind( &setUseColour, "yes|no" ); - - cli["--libidentify"] - .describe( "report name and version according to libidentify standard" ) - .bind( &ConfigData::libIdentify ); - - cli["--wait-for-keypress"] - .describe( "waits for a keypress before exiting" ) - .bind( &setWaitForKeypress, "start|exit|both" ); - - return cli; - } - -} // end namespace Catch - -// #included from: internal/catch_list.hpp -#define TWOBLUECUBES_CATCH_LIST_HPP_INCLUDED - -// #included from: catch_text.h -#define TWOBLUECUBES_CATCH_TEXT_H_INCLUDED - -#define TBC_TEXT_FORMAT_CONSOLE_WIDTH CATCH_CONFIG_CONSOLE_WIDTH - -#define CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE Catch -// #included from: ../external/tbc_text_format.h -// Only use header guard if we are not using an outer namespace -#ifndef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE -# ifdef TWOBLUECUBES_TEXT_FORMAT_H_INCLUDED -# ifndef TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED -# define TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED -# endif -# else -# define TWOBLUECUBES_TEXT_FORMAT_H_INCLUDED -# endif -#endif -#ifndef TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED -#include -#include -#include - -// Use optional outer namespace -#ifdef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE -namespace CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE { -#endif - -namespace Tbc { - -#ifdef TBC_TEXT_FORMAT_CONSOLE_WIDTH - const unsigned int consoleWidth = TBC_TEXT_FORMAT_CONSOLE_WIDTH; -#else - const unsigned int consoleWidth = 80; -#endif - - struct TextAttributes { - TextAttributes() - : initialIndent( std::string::npos ), - indent( 0 ), - width( consoleWidth-1 ) - {} - - TextAttributes& setInitialIndent( std::size_t _value ) { initialIndent = _value; return *this; } - TextAttributes& setIndent( std::size_t _value ) { indent = _value; return *this; } - TextAttributes& setWidth( std::size_t _value ) { width = _value; return *this; } - - std::size_t initialIndent; // indent of first line, or npos - std::size_t indent; // indent of subsequent lines, or all if initialIndent is npos - std::size_t width; // maximum width of text, including indent. Longer text will wrap - }; - - class Text { - public: - Text( std::string const& _str, TextAttributes const& _attr = TextAttributes() ) - : attr( _attr ) - { - const std::string wrappableBeforeChars = "[({<\t"; - const std::string wrappableAfterChars = "])}>-,./|\\"; - const std::string wrappableInsteadOfChars = " \n\r"; - std::string indent = _attr.initialIndent != std::string::npos - ? std::string( _attr.initialIndent, ' ' ) - : std::string( _attr.indent, ' ' ); - - typedef std::string::const_iterator iterator; - iterator it = _str.begin(); - const iterator strEnd = _str.end(); - - while( it != strEnd ) { - - if( lines.size() >= 1000 ) { - lines.push_back( "... message truncated due to excessive size" ); - return; - } - - std::string suffix; - std::size_t width = (std::min)( static_cast( strEnd-it ), _attr.width-static_cast( indent.size() ) ); - iterator itEnd = it+width; - iterator itNext = _str.end(); - - iterator itNewLine = std::find( it, itEnd, '\n' ); - if( itNewLine != itEnd ) - itEnd = itNewLine; - - if( itEnd != strEnd ) { - bool foundWrapPoint = false; - iterator findIt = itEnd; - do { - if( wrappableAfterChars.find( *findIt ) != std::string::npos && findIt != itEnd ) { - itEnd = findIt+1; - itNext = findIt+1; - foundWrapPoint = true; - } - else if( findIt > it && wrappableBeforeChars.find( *findIt ) != std::string::npos ) { - itEnd = findIt; - itNext = findIt; - foundWrapPoint = true; - } - else if( wrappableInsteadOfChars.find( *findIt ) != std::string::npos ) { - itNext = findIt+1; - itEnd = findIt; - foundWrapPoint = true; - } - if( findIt == it ) - break; - else - --findIt; - } - while( !foundWrapPoint ); - - if( !foundWrapPoint ) { - // No good wrap char, so we'll break mid word and add a hyphen - --itEnd; - itNext = itEnd; - suffix = "-"; - } - else { - while( itEnd > it && wrappableInsteadOfChars.find( *(itEnd-1) ) != std::string::npos ) - --itEnd; - } - } - lines.push_back( indent + std::string( it, itEnd ) + suffix ); - - if( indent.size() != _attr.indent ) - indent = std::string( _attr.indent, ' ' ); - it = itNext; - } - } - - typedef std::vector::const_iterator const_iterator; - - const_iterator begin() const { return lines.begin(); } - const_iterator end() const { return lines.end(); } - std::string const& last() const { return lines.back(); } - std::size_t size() const { return lines.size(); } - std::string const& operator[]( std::size_t _index ) const { return lines[_index]; } - std::string toString() const { - std::ostringstream oss; - oss << *this; - return oss.str(); - } - - inline friend std::ostream& operator << ( std::ostream& _stream, Text const& _text ) { - for( Text::const_iterator it = _text.begin(), itEnd = _text.end(); - it != itEnd; ++it ) { - if( it != _text.begin() ) - _stream << "\n"; - _stream << *it; - } - return _stream; - } - - private: - std::string str; - TextAttributes attr; - std::vector lines; - }; - -} // end namespace Tbc - -#ifdef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE -} // end outer namespace -#endif - -#endif // TWOBLUECUBES_TEXT_FORMAT_H_ALREADY_INCLUDED -#undef CLICHE_TBC_TEXT_FORMAT_OUTER_NAMESPACE - -namespace Catch { - using Tbc::Text; - using Tbc::TextAttributes; -} - -// #included from: catch_console_colour.hpp -#define TWOBLUECUBES_CATCH_CONSOLE_COLOUR_HPP_INCLUDED - -namespace Catch { - - struct Colour { - enum Code { - None = 0, - - White, - Red, - Green, - Blue, - Cyan, - Yellow, - Grey, - - Bright = 0x10, - - BrightRed = Bright | Red, - BrightGreen = Bright | Green, - LightGrey = Bright | Grey, - BrightWhite = Bright | White, - - // By intention - FileName = LightGrey, - Warning = Yellow, - ResultError = BrightRed, - ResultSuccess = BrightGreen, - ResultExpectedFailure = Warning, - - Error = BrightRed, - Success = Green, - - OriginalExpression = Cyan, - ReconstructedExpression = Yellow, - - SecondaryText = LightGrey, - Headers = White - }; - - // Use constructed object for RAII guard - Colour( Code _colourCode ); - Colour( Colour const& other ); - ~Colour(); - - // Use static method for one-shot changes - static void use( Code _colourCode ); - - private: - bool m_moved; - }; - - inline std::ostream& operator << ( std::ostream& os, Colour const& ) { return os; } - -} // end namespace Catch - -// #included from: catch_interfaces_reporter.h -#define TWOBLUECUBES_CATCH_INTERFACES_REPORTER_H_INCLUDED - -#include -#include -#include - -namespace Catch -{ - struct ReporterConfig { - explicit ReporterConfig( Ptr const& _fullConfig ) - : m_stream( &_fullConfig->stream() ), m_fullConfig( _fullConfig ) {} - - ReporterConfig( Ptr const& _fullConfig, std::ostream& _stream ) - : m_stream( &_stream ), m_fullConfig( _fullConfig ) {} - - std::ostream& stream() const { return *m_stream; } - Ptr fullConfig() const { return m_fullConfig; } - - private: - std::ostream* m_stream; - Ptr m_fullConfig; - }; - - struct ReporterPreferences { - ReporterPreferences() - : shouldRedirectStdOut( false ) - {} - - bool shouldRedirectStdOut; - }; - - template - struct LazyStat : Option { - LazyStat() : used( false ) {} - LazyStat& operator=( T const& _value ) { - Option::operator=( _value ); - used = false; - return *this; - } - void reset() { - Option::reset(); - used = false; - } - bool used; - }; - - struct TestRunInfo { - TestRunInfo( std::string const& _name ) : name( _name ) {} - std::string name; - }; - struct GroupInfo { - GroupInfo( std::string const& _name, - std::size_t _groupIndex, - std::size_t _groupsCount ) - : name( _name ), - groupIndex( _groupIndex ), - groupsCounts( _groupsCount ) - {} - - std::string name; - std::size_t groupIndex; - std::size_t groupsCounts; - }; - - struct AssertionStats { - AssertionStats( AssertionResult const& _assertionResult, - std::vector const& _infoMessages, - Totals const& _totals ) - : assertionResult( _assertionResult ), - infoMessages( _infoMessages ), - totals( _totals ) - { - if( assertionResult.hasMessage() ) { - // Copy message into messages list. - // !TBD This should have been done earlier, somewhere - MessageBuilder builder( assertionResult.getTestMacroName(), assertionResult.getSourceInfo(), assertionResult.getResultType() ); - builder << assertionResult.getMessage(); - builder.m_info.message = builder.m_stream.str(); - - infoMessages.push_back( builder.m_info ); - } - } - virtual ~AssertionStats(); - -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - AssertionStats( AssertionStats const& ) = default; - AssertionStats( AssertionStats && ) = default; - AssertionStats& operator = ( AssertionStats const& ) = default; - AssertionStats& operator = ( AssertionStats && ) = default; -# endif - - AssertionResult assertionResult; - std::vector infoMessages; - Totals totals; - }; - - struct SectionStats { - SectionStats( SectionInfo const& _sectionInfo, - Counts const& _assertions, - double _durationInSeconds, - bool _missingAssertions ) - : sectionInfo( _sectionInfo ), - assertions( _assertions ), - durationInSeconds( _durationInSeconds ), - missingAssertions( _missingAssertions ) - {} - virtual ~SectionStats(); -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - SectionStats( SectionStats const& ) = default; - SectionStats( SectionStats && ) = default; - SectionStats& operator = ( SectionStats const& ) = default; - SectionStats& operator = ( SectionStats && ) = default; -# endif - - SectionInfo sectionInfo; - Counts assertions; - double durationInSeconds; - bool missingAssertions; - }; - - struct TestCaseStats { - TestCaseStats( TestCaseInfo const& _testInfo, - Totals const& _totals, - std::string const& _stdOut, - std::string const& _stdErr, - bool _aborting ) - : testInfo( _testInfo ), - totals( _totals ), - stdOut( _stdOut ), - stdErr( _stdErr ), - aborting( _aborting ) - {} - virtual ~TestCaseStats(); - -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - TestCaseStats( TestCaseStats const& ) = default; - TestCaseStats( TestCaseStats && ) = default; - TestCaseStats& operator = ( TestCaseStats const& ) = default; - TestCaseStats& operator = ( TestCaseStats && ) = default; -# endif - - TestCaseInfo testInfo; - Totals totals; - std::string stdOut; - std::string stdErr; - bool aborting; - }; - - struct TestGroupStats { - TestGroupStats( GroupInfo const& _groupInfo, - Totals const& _totals, - bool _aborting ) - : groupInfo( _groupInfo ), - totals( _totals ), - aborting( _aborting ) - {} - TestGroupStats( GroupInfo const& _groupInfo ) - : groupInfo( _groupInfo ), - aborting( false ) - {} - virtual ~TestGroupStats(); - -# ifdef CATCH_CONFIG_CPP11_GENERATED_METHODS - TestGroupStats( TestGroupStats const& ) = default; - TestGroupStats( TestGroupStats && ) = default; - TestGroupStats& operator = ( TestGroupStats const& ) = default; - TestGroupStats& operator = ( TestGroupStats && ) = default; -# endif - - GroupInfo groupInfo; - Totals totals; - bool aborting; - }; - - struct TestRunStats { - TestRunStats( TestRunInfo const& _runInfo, - Totals const& _totals, - bool _aborting ) - : runInfo( _runInfo ), - totals( _totals ), - aborting( _aborting ) - {} - virtual ~TestRunStats(); - -# ifndef CATCH_CONFIG_CPP11_GENERATED_METHODS - TestRunStats( TestRunStats const& _other ) - : runInfo( _other.runInfo ), - totals( _other.totals ), - aborting( _other.aborting ) - {} -# else - TestRunStats( TestRunStats const& ) = default; - TestRunStats( TestRunStats && ) = default; - TestRunStats& operator = ( TestRunStats const& ) = default; - TestRunStats& operator = ( TestRunStats && ) = default; -# endif - - TestRunInfo runInfo; - Totals totals; - bool aborting; - }; - - class MultipleReporters; - - struct IStreamingReporter : IShared { - virtual ~IStreamingReporter(); - - // Implementing class must also provide the following static method: - // static std::string getDescription(); - - virtual ReporterPreferences getPreferences() const = 0; - - virtual void noMatchingTestCases( std::string const& spec ) = 0; - - virtual void testRunStarting( TestRunInfo const& testRunInfo ) = 0; - virtual void testGroupStarting( GroupInfo const& groupInfo ) = 0; - - virtual void testCaseStarting( TestCaseInfo const& testInfo ) = 0; - virtual void sectionStarting( SectionInfo const& sectionInfo ) = 0; - - virtual void assertionStarting( AssertionInfo const& assertionInfo ) = 0; - - // The return value indicates if the messages buffer should be cleared: - virtual bool assertionEnded( AssertionStats const& assertionStats ) = 0; - - virtual void sectionEnded( SectionStats const& sectionStats ) = 0; - virtual void testCaseEnded( TestCaseStats const& testCaseStats ) = 0; - virtual void testGroupEnded( TestGroupStats const& testGroupStats ) = 0; - virtual void testRunEnded( TestRunStats const& testRunStats ) = 0; - - virtual void skipTest( TestCaseInfo const& testInfo ) = 0; - - virtual MultipleReporters* tryAsMulti() { return CATCH_NULL; } - }; - - struct IReporterFactory : IShared { - virtual ~IReporterFactory(); - virtual IStreamingReporter* create( ReporterConfig const& config ) const = 0; - virtual std::string getDescription() const = 0; - }; - - struct IReporterRegistry { - typedef std::map > FactoryMap; - typedef std::vector > Listeners; - - virtual ~IReporterRegistry(); - virtual IStreamingReporter* create( std::string const& name, Ptr const& config ) const = 0; - virtual FactoryMap const& getFactories() const = 0; - virtual Listeners const& getListeners() const = 0; - }; - - Ptr addReporter( Ptr const& existingReporter, Ptr const& additionalReporter ); - -} - -#include -#include - -namespace Catch { - - inline std::size_t listTests( Config const& config ) { - - TestSpec testSpec = config.testSpec(); - if( config.testSpec().hasFilters() ) - Catch::cout() << "Matching test cases:\n"; - else { - Catch::cout() << "All available test cases:\n"; - testSpec = TestSpecParser( ITagAliasRegistry::get() ).parse( "*" ).testSpec(); - } - - std::size_t matchedTests = 0; - TextAttributes nameAttr, descAttr, tagsAttr; - nameAttr.setInitialIndent( 2 ).setIndent( 4 ); - descAttr.setIndent( 4 ); - tagsAttr.setIndent( 6 ); - - std::vector matchedTestCases = filterTests( getAllTestCasesSorted( config ), testSpec, config ); - for( std::vector::const_iterator it = matchedTestCases.begin(), itEnd = matchedTestCases.end(); - it != itEnd; - ++it ) { - matchedTests++; - TestCaseInfo const& testCaseInfo = it->getTestCaseInfo(); - Colour::Code colour = testCaseInfo.isHidden() - ? Colour::SecondaryText - : Colour::None; - Colour colourGuard( colour ); - - Catch::cout() << Text( testCaseInfo.name, nameAttr ) << std::endl; - if( config.listExtraInfo() ) { - Catch::cout() << " " << testCaseInfo.lineInfo << std::endl; - std::string description = testCaseInfo.description; - if( description.empty() ) - description = "(NO DESCRIPTION)"; - Catch::cout() << Text( description, descAttr ) << std::endl; - } - if( !testCaseInfo.tags.empty() ) - Catch::cout() << Text( testCaseInfo.tagsAsString, tagsAttr ) << std::endl; - } - - if( !config.testSpec().hasFilters() ) - Catch::cout() << pluralise( matchedTests, "test case" ) << '\n' << std::endl; - else - Catch::cout() << pluralise( matchedTests, "matching test case" ) << '\n' << std::endl; - return matchedTests; - } - - inline std::size_t listTestsNamesOnly( Config const& config ) { - TestSpec testSpec = config.testSpec(); - if( !config.testSpec().hasFilters() ) - testSpec = TestSpecParser( ITagAliasRegistry::get() ).parse( "*" ).testSpec(); - std::size_t matchedTests = 0; - std::vector matchedTestCases = filterTests( getAllTestCasesSorted( config ), testSpec, config ); - for( std::vector::const_iterator it = matchedTestCases.begin(), itEnd = matchedTestCases.end(); - it != itEnd; - ++it ) { - matchedTests++; - TestCaseInfo const& testCaseInfo = it->getTestCaseInfo(); - if( startsWith( testCaseInfo.name, '#' ) ) - Catch::cout() << '"' << testCaseInfo.name << '"'; - else - Catch::cout() << testCaseInfo.name; - if ( config.listExtraInfo() ) - Catch::cout() << "\t@" << testCaseInfo.lineInfo; - Catch::cout() << std::endl; - } - return matchedTests; - } - - struct TagInfo { - TagInfo() : count ( 0 ) {} - void add( std::string const& spelling ) { - ++count; - spellings.insert( spelling ); - } - std::string all() const { - std::string out; - for( std::set::const_iterator it = spellings.begin(), itEnd = spellings.end(); - it != itEnd; - ++it ) - out += "[" + *it + "]"; - return out; - } - std::set spellings; - std::size_t count; - }; - - inline std::size_t listTags( Config const& config ) { - TestSpec testSpec = config.testSpec(); - if( config.testSpec().hasFilters() ) - Catch::cout() << "Tags for matching test cases:\n"; - else { - Catch::cout() << "All available tags:\n"; - testSpec = TestSpecParser( ITagAliasRegistry::get() ).parse( "*" ).testSpec(); - } - - std::map tagCounts; - - std::vector matchedTestCases = filterTests( getAllTestCasesSorted( config ), testSpec, config ); - for( std::vector::const_iterator it = matchedTestCases.begin(), itEnd = matchedTestCases.end(); - it != itEnd; - ++it ) { - for( std::set::const_iterator tagIt = it->getTestCaseInfo().tags.begin(), - tagItEnd = it->getTestCaseInfo().tags.end(); - tagIt != tagItEnd; - ++tagIt ) { - std::string tagName = *tagIt; - std::string lcaseTagName = toLower( tagName ); - std::map::iterator countIt = tagCounts.find( lcaseTagName ); - if( countIt == tagCounts.end() ) - countIt = tagCounts.insert( std::make_pair( lcaseTagName, TagInfo() ) ).first; - countIt->second.add( tagName ); - } - } - - for( std::map::const_iterator countIt = tagCounts.begin(), - countItEnd = tagCounts.end(); - countIt != countItEnd; - ++countIt ) { - std::ostringstream oss; - oss << " " << std::setw(2) << countIt->second.count << " "; - Text wrapper( countIt->second.all(), TextAttributes() - .setInitialIndent( 0 ) - .setIndent( oss.str().size() ) - .setWidth( CATCH_CONFIG_CONSOLE_WIDTH-10 ) ); - Catch::cout() << oss.str() << wrapper << '\n'; - } - Catch::cout() << pluralise( tagCounts.size(), "tag" ) << '\n' << std::endl; - return tagCounts.size(); - } - - inline std::size_t listReporters( Config const& /*config*/ ) { - Catch::cout() << "Available reporters:\n"; - IReporterRegistry::FactoryMap const& factories = getRegistryHub().getReporterRegistry().getFactories(); - IReporterRegistry::FactoryMap::const_iterator itBegin = factories.begin(), itEnd = factories.end(), it; - std::size_t maxNameLen = 0; - for(it = itBegin; it != itEnd; ++it ) - maxNameLen = (std::max)( maxNameLen, it->first.size() ); - - for(it = itBegin; it != itEnd; ++it ) { - Text wrapper( it->second->getDescription(), TextAttributes() - .setInitialIndent( 0 ) - .setIndent( 7+maxNameLen ) - .setWidth( CATCH_CONFIG_CONSOLE_WIDTH - maxNameLen-8 ) ); - Catch::cout() << " " - << it->first - << ':' - << std::string( maxNameLen - it->first.size() + 2, ' ' ) - << wrapper << '\n'; - } - Catch::cout() << std::endl; - return factories.size(); - } - - inline Option list( Config const& config ) { - Option listedCount; - if( config.listTests() || ( config.listExtraInfo() && !config.listTestNamesOnly() ) ) - listedCount = listedCount.valueOr(0) + listTests( config ); - if( config.listTestNamesOnly() ) - listedCount = listedCount.valueOr(0) + listTestsNamesOnly( config ); - if( config.listTags() ) - listedCount = listedCount.valueOr(0) + listTags( config ); - if( config.listReporters() ) - listedCount = listedCount.valueOr(0) + listReporters( config ); - return listedCount; - } - -} // end namespace Catch - -// #included from: internal/catch_run_context.hpp -#define TWOBLUECUBES_CATCH_RUNNER_IMPL_HPP_INCLUDED - -// #included from: catch_test_case_tracker.hpp -#define TWOBLUECUBES_CATCH_TEST_CASE_TRACKER_HPP_INCLUDED - -#include -#include -#include -#include -#include - -CATCH_INTERNAL_SUPPRESS_ETD_WARNINGS - -namespace Catch { -namespace TestCaseTracking { - - struct NameAndLocation { - std::string name; - SourceLineInfo location; - - NameAndLocation( std::string const& _name, SourceLineInfo const& _location ) - : name( _name ), - location( _location ) - {} - }; - - struct ITracker : SharedImpl<> { - virtual ~ITracker(); - - // static queries - virtual NameAndLocation const& nameAndLocation() const = 0; - - // dynamic queries - virtual bool isComplete() const = 0; // Successfully completed or failed - virtual bool isSuccessfullyCompleted() const = 0; - virtual bool isOpen() const = 0; // Started but not complete - virtual bool hasChildren() const = 0; - - virtual ITracker& parent() = 0; - - // actions - virtual void close() = 0; // Successfully complete - virtual void fail() = 0; - virtual void markAsNeedingAnotherRun() = 0; - - virtual void addChild( Ptr const& child ) = 0; - virtual ITracker* findChild( NameAndLocation const& nameAndLocation ) = 0; - virtual void openChild() = 0; - - // Debug/ checking - virtual bool isSectionTracker() const = 0; - virtual bool isIndexTracker() const = 0; - }; - - class TrackerContext { - - enum RunState { - NotStarted, - Executing, - CompletedCycle - }; - - Ptr m_rootTracker; - ITracker* m_currentTracker; - RunState m_runState; - - public: - - static TrackerContext& instance() { - static TrackerContext s_instance; - return s_instance; - } - - TrackerContext() - : m_currentTracker( CATCH_NULL ), - m_runState( NotStarted ) - {} - - ITracker& startRun(); - - void endRun() { - m_rootTracker.reset(); - m_currentTracker = CATCH_NULL; - m_runState = NotStarted; - } - - void startCycle() { - m_currentTracker = m_rootTracker.get(); - m_runState = Executing; - } - void completeCycle() { - m_runState = CompletedCycle; - } - - bool completedCycle() const { - return m_runState == CompletedCycle; - } - ITracker& currentTracker() { - return *m_currentTracker; - } - void setCurrentTracker( ITracker* tracker ) { - m_currentTracker = tracker; - } - }; - - class TrackerBase : public ITracker { - protected: - enum CycleState { - NotStarted, - Executing, - ExecutingChildren, - NeedsAnotherRun, - CompletedSuccessfully, - Failed - }; - class TrackerHasName { - NameAndLocation m_nameAndLocation; - public: - TrackerHasName( NameAndLocation const& nameAndLocation ) : m_nameAndLocation( nameAndLocation ) {} - bool operator ()( Ptr const& tracker ) { - return - tracker->nameAndLocation().name == m_nameAndLocation.name && - tracker->nameAndLocation().location == m_nameAndLocation.location; - } - }; - typedef std::vector > Children; - NameAndLocation m_nameAndLocation; - TrackerContext& m_ctx; - ITracker* m_parent; - Children m_children; - CycleState m_runState; - public: - TrackerBase( NameAndLocation const& nameAndLocation, TrackerContext& ctx, ITracker* parent ) - : m_nameAndLocation( nameAndLocation ), - m_ctx( ctx ), - m_parent( parent ), - m_runState( NotStarted ) - {} - virtual ~TrackerBase(); - - virtual NameAndLocation const& nameAndLocation() const CATCH_OVERRIDE { - return m_nameAndLocation; - } - virtual bool isComplete() const CATCH_OVERRIDE { - return m_runState == CompletedSuccessfully || m_runState == Failed; - } - virtual bool isSuccessfullyCompleted() const CATCH_OVERRIDE { - return m_runState == CompletedSuccessfully; - } - virtual bool isOpen() const CATCH_OVERRIDE { - return m_runState != NotStarted && !isComplete(); - } - virtual bool hasChildren() const CATCH_OVERRIDE { - return !m_children.empty(); - } - - virtual void addChild( Ptr const& child ) CATCH_OVERRIDE { - m_children.push_back( child ); - } - - virtual ITracker* findChild( NameAndLocation const& nameAndLocation ) CATCH_OVERRIDE { - Children::const_iterator it = std::find_if( m_children.begin(), m_children.end(), TrackerHasName( nameAndLocation ) ); - return( it != m_children.end() ) - ? it->get() - : CATCH_NULL; - } - virtual ITracker& parent() CATCH_OVERRIDE { - assert( m_parent ); // Should always be non-null except for root - return *m_parent; - } - - virtual void openChild() CATCH_OVERRIDE { - if( m_runState != ExecutingChildren ) { - m_runState = ExecutingChildren; - if( m_parent ) - m_parent->openChild(); - } - } - - virtual bool isSectionTracker() const CATCH_OVERRIDE { return false; } - virtual bool isIndexTracker() const CATCH_OVERRIDE { return false; } - - void open() { - m_runState = Executing; - moveToThis(); - if( m_parent ) - m_parent->openChild(); - } - - virtual void close() CATCH_OVERRIDE { - - // Close any still open children (e.g. generators) - while( &m_ctx.currentTracker() != this ) - m_ctx.currentTracker().close(); - - switch( m_runState ) { - case NotStarted: - case CompletedSuccessfully: - case Failed: - throw std::logic_error( "Illogical state" ); - - case NeedsAnotherRun: - break;; - - case Executing: - m_runState = CompletedSuccessfully; - break; - case ExecutingChildren: - if( m_children.empty() || m_children.back()->isComplete() ) - m_runState = CompletedSuccessfully; - break; - - default: - throw std::logic_error( "Unexpected state" ); - } - moveToParent(); - m_ctx.completeCycle(); - } - virtual void fail() CATCH_OVERRIDE { - m_runState = Failed; - if( m_parent ) - m_parent->markAsNeedingAnotherRun(); - moveToParent(); - m_ctx.completeCycle(); - } - virtual void markAsNeedingAnotherRun() CATCH_OVERRIDE { - m_runState = NeedsAnotherRun; - } - private: - void moveToParent() { - assert( m_parent ); - m_ctx.setCurrentTracker( m_parent ); - } - void moveToThis() { - m_ctx.setCurrentTracker( this ); - } - }; - - class SectionTracker : public TrackerBase { - std::vector m_filters; - public: - SectionTracker( NameAndLocation const& nameAndLocation, TrackerContext& ctx, ITracker* parent ) - : TrackerBase( nameAndLocation, ctx, parent ) - { - if( parent ) { - while( !parent->isSectionTracker() ) - parent = &parent->parent(); - - SectionTracker& parentSection = static_cast( *parent ); - addNextFilters( parentSection.m_filters ); - } - } - virtual ~SectionTracker(); - - virtual bool isSectionTracker() const CATCH_OVERRIDE { return true; } - - static SectionTracker& acquire( TrackerContext& ctx, NameAndLocation const& nameAndLocation ) { - SectionTracker* section = CATCH_NULL; - - ITracker& currentTracker = ctx.currentTracker(); - if( ITracker* childTracker = currentTracker.findChild( nameAndLocation ) ) { - assert( childTracker ); - assert( childTracker->isSectionTracker() ); - section = static_cast( childTracker ); - } - else { - section = new SectionTracker( nameAndLocation, ctx, ¤tTracker ); - currentTracker.addChild( section ); - } - if( !ctx.completedCycle() ) - section->tryOpen(); - return *section; - } - - void tryOpen() { - if( !isComplete() && (m_filters.empty() || m_filters[0].empty() || m_filters[0] == m_nameAndLocation.name ) ) - open(); - } - - void addInitialFilters( std::vector const& filters ) { - if( !filters.empty() ) { - m_filters.push_back(""); // Root - should never be consulted - m_filters.push_back(""); // Test Case - not a section filter - m_filters.insert( m_filters.end(), filters.begin(), filters.end() ); - } - } - void addNextFilters( std::vector const& filters ) { - if( filters.size() > 1 ) - m_filters.insert( m_filters.end(), ++filters.begin(), filters.end() ); - } - }; - - class IndexTracker : public TrackerBase { - int m_size; - int m_index; - public: - IndexTracker( NameAndLocation const& nameAndLocation, TrackerContext& ctx, ITracker* parent, int size ) - : TrackerBase( nameAndLocation, ctx, parent ), - m_size( size ), - m_index( -1 ) - {} - virtual ~IndexTracker(); - - virtual bool isIndexTracker() const CATCH_OVERRIDE { return true; } - - static IndexTracker& acquire( TrackerContext& ctx, NameAndLocation const& nameAndLocation, int size ) { - IndexTracker* tracker = CATCH_NULL; - - ITracker& currentTracker = ctx.currentTracker(); - if( ITracker* childTracker = currentTracker.findChild( nameAndLocation ) ) { - assert( childTracker ); - assert( childTracker->isIndexTracker() ); - tracker = static_cast( childTracker ); - } - else { - tracker = new IndexTracker( nameAndLocation, ctx, ¤tTracker, size ); - currentTracker.addChild( tracker ); - } - - if( !ctx.completedCycle() && !tracker->isComplete() ) { - if( tracker->m_runState != ExecutingChildren && tracker->m_runState != NeedsAnotherRun ) - tracker->moveNext(); - tracker->open(); - } - - return *tracker; - } - - int index() const { return m_index; } - - void moveNext() { - m_index++; - m_children.clear(); - } - - virtual void close() CATCH_OVERRIDE { - TrackerBase::close(); - if( m_runState == CompletedSuccessfully && m_index < m_size-1 ) - m_runState = Executing; - } - }; - - inline ITracker& TrackerContext::startRun() { - m_rootTracker = new SectionTracker( NameAndLocation( "{root}", CATCH_INTERNAL_LINEINFO ), *this, CATCH_NULL ); - m_currentTracker = CATCH_NULL; - m_runState = Executing; - return *m_rootTracker; - } - -} // namespace TestCaseTracking - -using TestCaseTracking::ITracker; -using TestCaseTracking::TrackerContext; -using TestCaseTracking::SectionTracker; -using TestCaseTracking::IndexTracker; - -} // namespace Catch - -CATCH_INTERNAL_UNSUPPRESS_ETD_WARNINGS - -// #included from: catch_fatal_condition.hpp -#define TWOBLUECUBES_CATCH_FATAL_CONDITION_H_INCLUDED - -namespace Catch { - - // Report the error condition - inline void reportFatal( std::string const& message ) { - IContext& context = Catch::getCurrentContext(); - IResultCapture* resultCapture = context.getResultCapture(); - resultCapture->handleFatalErrorCondition( message ); - } - -} // namespace Catch - -#if defined ( CATCH_PLATFORM_WINDOWS ) ///////////////////////////////////////// -// #included from: catch_windows_h_proxy.h - -#define TWOBLUECUBES_CATCH_WINDOWS_H_PROXY_H_INCLUDED - -#ifdef CATCH_DEFINES_NOMINMAX -# define NOMINMAX -#endif -#ifdef CATCH_DEFINES_WIN32_LEAN_AND_MEAN -# define WIN32_LEAN_AND_MEAN -#endif - -#ifdef __AFXDLL -#include -#else -#include -#endif - -#ifdef CATCH_DEFINES_NOMINMAX -# undef NOMINMAX -#endif -#ifdef CATCH_DEFINES_WIN32_LEAN_AND_MEAN -# undef WIN32_LEAN_AND_MEAN -#endif - - -# if !defined ( CATCH_CONFIG_WINDOWS_SEH ) - -namespace Catch { - struct FatalConditionHandler { - void reset() {} - }; -} - -# else // CATCH_CONFIG_WINDOWS_SEH is defined - -namespace Catch { - - struct SignalDefs { DWORD id; const char* name; }; - extern SignalDefs signalDefs[]; - // There is no 1-1 mapping between signals and windows exceptions. - // Windows can easily distinguish between SO and SigSegV, - // but SigInt, SigTerm, etc are handled differently. - SignalDefs signalDefs[] = { - { EXCEPTION_ILLEGAL_INSTRUCTION, "SIGILL - Illegal instruction signal" }, - { EXCEPTION_STACK_OVERFLOW, "SIGSEGV - Stack overflow" }, - { EXCEPTION_ACCESS_VIOLATION, "SIGSEGV - Segmentation violation signal" }, - { EXCEPTION_INT_DIVIDE_BY_ZERO, "Divide by zero error" }, - }; - - struct FatalConditionHandler { - - static LONG CALLBACK handleVectoredException(PEXCEPTION_POINTERS ExceptionInfo) { - for (int i = 0; i < sizeof(signalDefs) / sizeof(SignalDefs); ++i) { - if (ExceptionInfo->ExceptionRecord->ExceptionCode == signalDefs[i].id) { - reportFatal(signalDefs[i].name); - } - } - // If its not an exception we care about, pass it along. - // This stops us from eating debugger breaks etc. - return EXCEPTION_CONTINUE_SEARCH; - } - - FatalConditionHandler() { - isSet = true; - // 32k seems enough for Catch to handle stack overflow, - // but the value was found experimentally, so there is no strong guarantee - guaranteeSize = 32 * 1024; - exceptionHandlerHandle = CATCH_NULL; - // Register as first handler in current chain - exceptionHandlerHandle = AddVectoredExceptionHandler(1, handleVectoredException); - // Pass in guarantee size to be filled - SetThreadStackGuarantee(&guaranteeSize); - } - - static void reset() { - if (isSet) { - // Unregister handler and restore the old guarantee - RemoveVectoredExceptionHandler(exceptionHandlerHandle); - SetThreadStackGuarantee(&guaranteeSize); - exceptionHandlerHandle = CATCH_NULL; - isSet = false; - } - } - - ~FatalConditionHandler() { - reset(); - } - private: - static bool isSet; - static ULONG guaranteeSize; - static PVOID exceptionHandlerHandle; - }; - - bool FatalConditionHandler::isSet = false; - ULONG FatalConditionHandler::guaranteeSize = 0; - PVOID FatalConditionHandler::exceptionHandlerHandle = CATCH_NULL; - -} // namespace Catch - -# endif // CATCH_CONFIG_WINDOWS_SEH - -#else // Not Windows - assumed to be POSIX compatible ////////////////////////// - -# if !defined(CATCH_CONFIG_POSIX_SIGNALS) - -namespace Catch { - struct FatalConditionHandler { - void reset() {} - }; -} - -# else // CATCH_CONFIG_POSIX_SIGNALS is defined - -#include - -namespace Catch { - - struct SignalDefs { - int id; - const char* name; - }; - extern SignalDefs signalDefs[]; - SignalDefs signalDefs[] = { - { SIGINT, "SIGINT - Terminal interrupt signal" }, - { SIGILL, "SIGILL - Illegal instruction signal" }, - { SIGFPE, "SIGFPE - Floating point error signal" }, - { SIGSEGV, "SIGSEGV - Segmentation violation signal" }, - { SIGTERM, "SIGTERM - Termination request signal" }, - { SIGABRT, "SIGABRT - Abort (abnormal termination) signal" } - }; - - struct FatalConditionHandler { - - static bool isSet; - static struct sigaction oldSigActions [sizeof(signalDefs)/sizeof(SignalDefs)]; - static stack_t oldSigStack; - static char altStackMem[SIGSTKSZ]; - - static void handleSignal( int sig ) { - std::string name = ""; - for (std::size_t i = 0; i < sizeof(signalDefs) / sizeof(SignalDefs); ++i) { - SignalDefs &def = signalDefs[i]; - if (sig == def.id) { - name = def.name; - break; - } - } - reset(); - reportFatal(name); - raise( sig ); - } - - FatalConditionHandler() { - isSet = true; - stack_t sigStack; - sigStack.ss_sp = altStackMem; - sigStack.ss_size = SIGSTKSZ; - sigStack.ss_flags = 0; - sigaltstack(&sigStack, &oldSigStack); - struct sigaction sa = { 0 }; - - sa.sa_handler = handleSignal; - sa.sa_flags = SA_ONSTACK; - for (std::size_t i = 0; i < sizeof(signalDefs)/sizeof(SignalDefs); ++i) { - sigaction(signalDefs[i].id, &sa, &oldSigActions[i]); - } - } - - ~FatalConditionHandler() { - reset(); - } - static void reset() { - if( isSet ) { - // Set signals back to previous values -- hopefully nobody overwrote them in the meantime - for( std::size_t i = 0; i < sizeof(signalDefs)/sizeof(SignalDefs); ++i ) { - sigaction(signalDefs[i].id, &oldSigActions[i], CATCH_NULL); - } - // Return the old stack - sigaltstack(&oldSigStack, CATCH_NULL); - isSet = false; - } - } - }; - - bool FatalConditionHandler::isSet = false; - struct sigaction FatalConditionHandler::oldSigActions[sizeof(signalDefs)/sizeof(SignalDefs)] = {}; - stack_t FatalConditionHandler::oldSigStack = {}; - char FatalConditionHandler::altStackMem[SIGSTKSZ] = {}; - -} // namespace Catch - -# endif // CATCH_CONFIG_POSIX_SIGNALS - -#endif // not Windows - -#include -#include -#include - -namespace Catch { - - class StreamRedirect { - - public: - StreamRedirect( std::ostream& stream, std::string& targetString ) - : m_stream( stream ), - m_prevBuf( stream.rdbuf() ), - m_targetString( targetString ) - { - stream.rdbuf( m_oss.rdbuf() ); - } - - ~StreamRedirect() { - m_targetString += m_oss.str(); - m_stream.rdbuf( m_prevBuf ); - } - - private: - std::ostream& m_stream; - std::streambuf* m_prevBuf; - std::ostringstream m_oss; - std::string& m_targetString; - }; - - // StdErr has two constituent streams in C++, std::cerr and std::clog - // This means that we need to redirect 2 streams into 1 to keep proper - // order of writes and cannot use StreamRedirect on its own - class StdErrRedirect { - public: - StdErrRedirect(std::string& targetString) - :m_cerrBuf( cerr().rdbuf() ), m_clogBuf(clog().rdbuf()), - m_targetString(targetString){ - cerr().rdbuf(m_oss.rdbuf()); - clog().rdbuf(m_oss.rdbuf()); - } - ~StdErrRedirect() { - m_targetString += m_oss.str(); - cerr().rdbuf(m_cerrBuf); - clog().rdbuf(m_clogBuf); - } - private: - std::streambuf* m_cerrBuf; - std::streambuf* m_clogBuf; - std::ostringstream m_oss; - std::string& m_targetString; - }; - - /////////////////////////////////////////////////////////////////////////// - - class RunContext : public IResultCapture, public IRunner { - - RunContext( RunContext const& ); - void operator =( RunContext const& ); - - public: - - explicit RunContext( Ptr const& _config, Ptr const& reporter ) - : m_runInfo( _config->name() ), - m_context( getCurrentMutableContext() ), - m_activeTestCase( CATCH_NULL ), - m_config( _config ), - m_reporter( reporter ), - m_shouldReportUnexpected ( true ) - { - m_context.setRunner( this ); - m_context.setConfig( m_config ); - m_context.setResultCapture( this ); - m_reporter->testRunStarting( m_runInfo ); - } - - virtual ~RunContext() { - m_reporter->testRunEnded( TestRunStats( m_runInfo, m_totals, aborting() ) ); - } - - void testGroupStarting( std::string const& testSpec, std::size_t groupIndex, std::size_t groupsCount ) { - m_reporter->testGroupStarting( GroupInfo( testSpec, groupIndex, groupsCount ) ); - } - void testGroupEnded( std::string const& testSpec, Totals const& totals, std::size_t groupIndex, std::size_t groupsCount ) { - m_reporter->testGroupEnded( TestGroupStats( GroupInfo( testSpec, groupIndex, groupsCount ), totals, aborting() ) ); - } - - Totals runTest( TestCase const& testCase ) { - Totals prevTotals = m_totals; - - std::string redirectedCout; - std::string redirectedCerr; - - TestCaseInfo testInfo = testCase.getTestCaseInfo(); - - m_reporter->testCaseStarting( testInfo ); - - m_activeTestCase = &testCase; - - do { - ITracker& rootTracker = m_trackerContext.startRun(); - assert( rootTracker.isSectionTracker() ); - static_cast( rootTracker ).addInitialFilters( m_config->getSectionsToRun() ); - do { - m_trackerContext.startCycle(); - m_testCaseTracker = &SectionTracker::acquire( m_trackerContext, TestCaseTracking::NameAndLocation( testInfo.name, testInfo.lineInfo ) ); - runCurrentTest( redirectedCout, redirectedCerr ); - } - while( !m_testCaseTracker->isSuccessfullyCompleted() && !aborting() ); - } - // !TBD: deprecated - this will be replaced by indexed trackers - while( getCurrentContext().advanceGeneratorsForCurrentTest() && !aborting() ); - - Totals deltaTotals = m_totals.delta( prevTotals ); - if( testInfo.expectedToFail() && deltaTotals.testCases.passed > 0 ) { - deltaTotals.assertions.failed++; - deltaTotals.testCases.passed--; - deltaTotals.testCases.failed++; - } - m_totals.testCases += deltaTotals.testCases; - m_reporter->testCaseEnded( TestCaseStats( testInfo, - deltaTotals, - redirectedCout, - redirectedCerr, - aborting() ) ); - - m_activeTestCase = CATCH_NULL; - m_testCaseTracker = CATCH_NULL; - - return deltaTotals; - } - - Ptr config() const { - return m_config; - } - - private: // IResultCapture - - virtual void assertionEnded( AssertionResult const& result ) { - if( result.getResultType() == ResultWas::Ok ) { - m_totals.assertions.passed++; - } - else if( !result.isOk() ) { - if( m_activeTestCase->getTestCaseInfo().okToFail() ) - m_totals.assertions.failedButOk++; - else - m_totals.assertions.failed++; - } - - // We have no use for the return value (whether messages should be cleared), because messages were made scoped - // and should be let to clear themselves out. - static_cast(m_reporter->assertionEnded(AssertionStats(result, m_messages, m_totals))); - - // Reset working state - m_lastAssertionInfo = AssertionInfo( "", m_lastAssertionInfo.lineInfo, "{Unknown expression after the reported line}" , m_lastAssertionInfo.resultDisposition ); - m_lastResult = result; - } - - virtual bool lastAssertionPassed() - { - return m_totals.assertions.passed == (m_prevPassed + 1); - } - - virtual void assertionPassed() - { - m_totals.assertions.passed++; - m_lastAssertionInfo.capturedExpression = "{Unknown expression after the reported line}"; - m_lastAssertionInfo.macroName = ""; - } - - virtual void assertionRun() - { - m_prevPassed = m_totals.assertions.passed; - } - - virtual bool sectionStarted ( - SectionInfo const& sectionInfo, - Counts& assertions - ) - { - ITracker& sectionTracker = SectionTracker::acquire( m_trackerContext, TestCaseTracking::NameAndLocation( sectionInfo.name, sectionInfo.lineInfo ) ); - if( !sectionTracker.isOpen() ) - return false; - m_activeSections.push_back( §ionTracker ); - - m_lastAssertionInfo.lineInfo = sectionInfo.lineInfo; - - m_reporter->sectionStarting( sectionInfo ); - - assertions = m_totals.assertions; - - return true; - } - bool testForMissingAssertions( Counts& assertions ) { - if( assertions.total() != 0 ) - return false; - if( !m_config->warnAboutMissingAssertions() ) - return false; - if( m_trackerContext.currentTracker().hasChildren() ) - return false; - m_totals.assertions.failed++; - assertions.failed++; - return true; - } - - virtual void sectionEnded( SectionEndInfo const& endInfo ) { - Counts assertions = m_totals.assertions - endInfo.prevAssertions; - bool missingAssertions = testForMissingAssertions( assertions ); - - if( !m_activeSections.empty() ) { - m_activeSections.back()->close(); - m_activeSections.pop_back(); - } - - m_reporter->sectionEnded( SectionStats( endInfo.sectionInfo, assertions, endInfo.durationInSeconds, missingAssertions ) ); - m_messages.clear(); - } - - virtual void sectionEndedEarly( SectionEndInfo const& endInfo ) { - if( m_unfinishedSections.empty() ) - m_activeSections.back()->fail(); - else - m_activeSections.back()->close(); - m_activeSections.pop_back(); - - m_unfinishedSections.push_back( endInfo ); - } - - virtual void pushScopedMessage( MessageInfo const& message ) { - m_messages.push_back( message ); - } - - virtual void popScopedMessage( MessageInfo const& message ) { - m_messages.erase( std::remove( m_messages.begin(), m_messages.end(), message ), m_messages.end() ); - } - - virtual std::string getCurrentTestName() const { - return m_activeTestCase - ? m_activeTestCase->getTestCaseInfo().name - : std::string(); - } - - virtual const AssertionResult* getLastResult() const { - return &m_lastResult; - } - - virtual void exceptionEarlyReported() { - m_shouldReportUnexpected = false; - } - - virtual void handleFatalErrorCondition( std::string const& message ) { - // Don't rebuild the result -- the stringification itself can cause more fatal errors - // Instead, fake a result data. - AssertionResultData tempResult; - tempResult.resultType = ResultWas::FatalErrorCondition; - tempResult.message = message; - AssertionResult result(m_lastAssertionInfo, tempResult); - - getResultCapture().assertionEnded(result); - - handleUnfinishedSections(); - - // Recreate section for test case (as we will lose the one that was in scope) - TestCaseInfo const& testCaseInfo = m_activeTestCase->getTestCaseInfo(); - SectionInfo testCaseSection( testCaseInfo.lineInfo, testCaseInfo.name, testCaseInfo.description ); - - Counts assertions; - assertions.failed = 1; - SectionStats testCaseSectionStats( testCaseSection, assertions, 0, false ); - m_reporter->sectionEnded( testCaseSectionStats ); - - TestCaseInfo testInfo = m_activeTestCase->getTestCaseInfo(); - - Totals deltaTotals; - deltaTotals.testCases.failed = 1; - deltaTotals.assertions.failed = 1; - m_reporter->testCaseEnded( TestCaseStats( testInfo, - deltaTotals, - std::string(), - std::string(), - false ) ); - m_totals.testCases.failed++; - testGroupEnded( std::string(), m_totals, 1, 1 ); - m_reporter->testRunEnded( TestRunStats( m_runInfo, m_totals, false ) ); - } - - public: - // !TBD We need to do this another way! - bool aborting() const { - return m_totals.assertions.failed == static_cast( m_config->abortAfter() ); - } - - private: - - void runCurrentTest( std::string& redirectedCout, std::string& redirectedCerr ) { - TestCaseInfo const& testCaseInfo = m_activeTestCase->getTestCaseInfo(); - SectionInfo testCaseSection( testCaseInfo.lineInfo, testCaseInfo.name, testCaseInfo.description ); - m_reporter->sectionStarting( testCaseSection ); - Counts prevAssertions = m_totals.assertions; - double duration = 0; - m_shouldReportUnexpected = true; - try { - m_lastAssertionInfo = AssertionInfo( "TEST_CASE", testCaseInfo.lineInfo, "", ResultDisposition::Normal ); - - seedRng( *m_config ); - - Timer timer; - timer.start(); - if( m_reporter->getPreferences().shouldRedirectStdOut ) { - StreamRedirect coutRedir( Catch::cout(), redirectedCout ); - StdErrRedirect errRedir( redirectedCerr ); - invokeActiveTestCase(); - } - else { - invokeActiveTestCase(); - } - duration = timer.getElapsedSeconds(); - } - catch( TestFailureException& ) { - // This just means the test was aborted due to failure - } - catch(...) { - // Under CATCH_CONFIG_FAST_COMPILE, unexpected exceptions under REQUIRE assertions - // are reported without translation at the point of origin. - if (m_shouldReportUnexpected) { - makeUnexpectedResultBuilder().useActiveException(); - } - } - m_testCaseTracker->close(); - handleUnfinishedSections(); - m_messages.clear(); - - Counts assertions = m_totals.assertions - prevAssertions; - bool missingAssertions = testForMissingAssertions( assertions ); - - SectionStats testCaseSectionStats( testCaseSection, assertions, duration, missingAssertions ); - m_reporter->sectionEnded( testCaseSectionStats ); - } - - void invokeActiveTestCase() { - FatalConditionHandler fatalConditionHandler; // Handle signals - m_activeTestCase->invoke(); - fatalConditionHandler.reset(); - } - - private: - - ResultBuilder makeUnexpectedResultBuilder() const { - return ResultBuilder( m_lastAssertionInfo.macroName, - m_lastAssertionInfo.lineInfo, - m_lastAssertionInfo.capturedExpression, - m_lastAssertionInfo.resultDisposition ); - } - - void handleUnfinishedSections() { - // If sections ended prematurely due to an exception we stored their - // infos here so we can tear them down outside the unwind process. - for( std::vector::const_reverse_iterator it = m_unfinishedSections.rbegin(), - itEnd = m_unfinishedSections.rend(); - it != itEnd; - ++it ) - sectionEnded( *it ); - m_unfinishedSections.clear(); - } - - TestRunInfo m_runInfo; - IMutableContext& m_context; - TestCase const* m_activeTestCase; - ITracker* m_testCaseTracker; - ITracker* m_currentSectionTracker; - AssertionResult m_lastResult; - - Ptr m_config; - Totals m_totals; - Ptr m_reporter; - std::vector m_messages; - AssertionInfo m_lastAssertionInfo; - std::vector m_unfinishedSections; - std::vector m_activeSections; - TrackerContext m_trackerContext; - size_t m_prevPassed; - bool m_shouldReportUnexpected; - }; - - IResultCapture& getResultCapture() { - if( IResultCapture* capture = getCurrentContext().getResultCapture() ) - return *capture; - else - throw std::logic_error( "No result capture instance" ); - } - -} // end namespace Catch - -// #included from: internal/catch_version.h -#define TWOBLUECUBES_CATCH_VERSION_H_INCLUDED - -namespace Catch { - - // Versioning information - struct Version { - Version( unsigned int _majorVersion, - unsigned int _minorVersion, - unsigned int _patchNumber, - char const * const _branchName, - unsigned int _buildNumber ); - - unsigned int const majorVersion; - unsigned int const minorVersion; - unsigned int const patchNumber; - - // buildNumber is only used if branchName is not null - char const * const branchName; - unsigned int const buildNumber; - - friend std::ostream& operator << ( std::ostream& os, Version const& version ); - - private: - void operator=( Version const& ); - }; - - inline Version libraryVersion(); -} - -#include -#include -#include - -namespace Catch { - - Ptr createReporter( std::string const& reporterName, Ptr const& config ) { - Ptr reporter = getRegistryHub().getReporterRegistry().create( reporterName, config.get() ); - if( !reporter ) { - std::ostringstream oss; - oss << "No reporter registered with name: '" << reporterName << "'"; - throw std::domain_error( oss.str() ); - } - return reporter; - } - -#if !defined(CATCH_CONFIG_DEFAULT_REPORTER) -#define CATCH_CONFIG_DEFAULT_REPORTER "console" -#endif - - Ptr makeReporter( Ptr const& config ) { - std::vector reporters = config->getReporterNames(); - if( reporters.empty() ) - reporters.push_back( CATCH_CONFIG_DEFAULT_REPORTER ); - - Ptr reporter; - for( std::vector::const_iterator it = reporters.begin(), itEnd = reporters.end(); - it != itEnd; - ++it ) - reporter = addReporter( reporter, createReporter( *it, config ) ); - return reporter; - } - Ptr addListeners( Ptr const& config, Ptr reporters ) { - IReporterRegistry::Listeners listeners = getRegistryHub().getReporterRegistry().getListeners(); - for( IReporterRegistry::Listeners::const_iterator it = listeners.begin(), itEnd = listeners.end(); - it != itEnd; - ++it ) - reporters = addReporter(reporters, (*it)->create( ReporterConfig( config ) ) ); - return reporters; - } - - Totals runTests( Ptr const& config ) { - - Ptr iconfig = config.get(); - - Ptr reporter = makeReporter( config ); - reporter = addListeners( iconfig, reporter ); - - RunContext context( iconfig, reporter ); - - Totals totals; - - context.testGroupStarting( config->name(), 1, 1 ); - - TestSpec testSpec = config->testSpec(); - if( !testSpec.hasFilters() ) - testSpec = TestSpecParser( ITagAliasRegistry::get() ).parse( "~[.]" ).testSpec(); // All not hidden tests - - std::vector const& allTestCases = getAllTestCasesSorted( *iconfig ); - for( std::vector::const_iterator it = allTestCases.begin(), itEnd = allTestCases.end(); - it != itEnd; - ++it ) { - if( !context.aborting() && matchTest( *it, testSpec, *iconfig ) ) - totals += context.runTest( *it ); - else - reporter->skipTest( *it ); - } - - context.testGroupEnded( iconfig->name(), totals, 1, 1 ); - return totals; - } - - void applyFilenamesAsTags( IConfig const& config ) { - std::vector const& tests = getAllTestCasesSorted( config ); - for(std::size_t i = 0; i < tests.size(); ++i ) { - TestCase& test = const_cast( tests[i] ); - std::set tags = test.tags; - - std::string filename = test.lineInfo.file; - std::string::size_type lastSlash = filename.find_last_of( "\\/" ); - if( lastSlash != std::string::npos ) - filename = filename.substr( lastSlash+1 ); - - std::string::size_type lastDot = filename.find_last_of( '.' ); - if( lastDot != std::string::npos ) - filename = filename.substr( 0, lastDot ); - - tags.insert( '#' + filename ); - setTags( test, tags ); - } - } - - class Session : NonCopyable { - static bool alreadyInstantiated; - - public: - - struct OnUnusedOptions { enum DoWhat { Ignore, Fail }; }; - - Session() - : m_cli( makeCommandLineParser() ) { - if( alreadyInstantiated ) { - std::string msg = "Only one instance of Catch::Session can ever be used"; - Catch::cerr() << msg << std::endl; - throw std::logic_error( msg ); - } - alreadyInstantiated = true; - } - ~Session() { - Catch::cleanUp(); - } - - void showHelp( std::string const& processName ) { - Catch::cout() << "\nCatch v" << libraryVersion() << "\n"; - - m_cli.usage( Catch::cout(), processName ); - Catch::cout() << "For more detail usage please see the project docs\n" << std::endl; - } - void libIdentify() { - Catch::cout() - << std::left << std::setw(16) << "description: " << "A Catch test executable\n" - << std::left << std::setw(16) << "category: " << "testframework\n" - << std::left << std::setw(16) << "framework: " << "Catch Test\n" - << std::left << std::setw(16) << "version: " << libraryVersion() << std::endl; - } - - int applyCommandLine( int argc, char const* const* const argv, OnUnusedOptions::DoWhat unusedOptionBehaviour = OnUnusedOptions::Fail ) { - try { - m_cli.setThrowOnUnrecognisedTokens( unusedOptionBehaviour == OnUnusedOptions::Fail ); - m_unusedTokens = m_cli.parseInto( Clara::argsToVector( argc, argv ), m_configData ); - if( m_configData.showHelp ) - showHelp( m_configData.processName ); - if( m_configData.libIdentify ) - libIdentify(); - m_config.reset(); - } - catch( std::exception& ex ) { - { - Colour colourGuard( Colour::Red ); - Catch::cerr() - << "\nError(s) in input:\n" - << Text( ex.what(), TextAttributes().setIndent(2) ) - << "\n\n"; - } - m_cli.usage( Catch::cout(), m_configData.processName ); - return (std::numeric_limits::max)(); - } - return 0; - } - - void useConfigData( ConfigData const& _configData ) { - m_configData = _configData; - m_config.reset(); - } - - int run( int argc, char const* const* const argv ) { - - int returnCode = applyCommandLine( argc, argv ); - if( returnCode == 0 ) - returnCode = run(); - return returnCode; - } - - #if defined(WIN32) && defined(UNICODE) - int run( int argc, wchar_t const* const* const argv ) { - - char **utf8Argv = new char *[ argc ]; - - for ( int i = 0; i < argc; ++i ) { - int bufSize = WideCharToMultiByte( CP_UTF8, 0, argv[i], -1, NULL, 0, NULL, NULL ); - - utf8Argv[ i ] = new char[ bufSize ]; - - WideCharToMultiByte( CP_UTF8, 0, argv[i], -1, utf8Argv[i], bufSize, NULL, NULL ); - } - - int returnCode = applyCommandLine( argc, utf8Argv ); - if( returnCode == 0 ) - returnCode = run(); - - for ( int i = 0; i < argc; ++i ) - delete [] utf8Argv[ i ]; - - delete [] utf8Argv; - - return returnCode; - } - #endif - - int run() { - if( ( m_configData.waitForKeypress & WaitForKeypress::BeforeStart ) != 0 ) { - Catch::cout() << "...waiting for enter/ return before starting" << std::endl; - static_cast(std::getchar()); - } - int exitCode = runInternal(); - if( ( m_configData.waitForKeypress & WaitForKeypress::BeforeExit ) != 0 ) { - Catch::cout() << "...waiting for enter/ return before exiting, with code: " << exitCode << std::endl; - static_cast(std::getchar()); - } - return exitCode; - } - - Clara::CommandLine const& cli() const { - return m_cli; - } - std::vector const& unusedTokens() const { - return m_unusedTokens; - } - ConfigData& configData() { - return m_configData; - } - Config& config() { - if( !m_config ) - m_config = new Config( m_configData ); - return *m_config; - } - private: - - int runInternal() { - if( m_configData.showHelp || m_configData.libIdentify ) - return 0; - - try - { - config(); // Force config to be constructed - - seedRng( *m_config ); - - if( m_configData.filenamesAsTags ) - applyFilenamesAsTags( *m_config ); - - // Handle list request - if( Option listed = list( config() ) ) - return static_cast( *listed ); - - return static_cast( runTests( m_config ).assertions.failed ); - } - catch( std::exception& ex ) { - Catch::cerr() << ex.what() << std::endl; - return (std::numeric_limits::max)(); - } - } - - Clara::CommandLine m_cli; - std::vector m_unusedTokens; - ConfigData m_configData; - Ptr m_config; - }; - - bool Session::alreadyInstantiated = false; - -} // end namespace Catch - -// #included from: catch_registry_hub.hpp -#define TWOBLUECUBES_CATCH_REGISTRY_HUB_HPP_INCLUDED - -// #included from: catch_test_case_registry_impl.hpp -#define TWOBLUECUBES_CATCH_TEST_CASE_REGISTRY_IMPL_HPP_INCLUDED - -#include -#include -#include -#include - -namespace Catch { - - struct RandomNumberGenerator { - typedef unsigned int result_type; - - result_type operator()( result_type n ) const { return std::rand() % n; } - -#ifdef CATCH_CONFIG_CPP11_SHUFFLE - static constexpr result_type (min)() { return 0; } - static constexpr result_type (max)() { return 1000000; } - result_type operator()() const { return std::rand() % (max)(); } -#endif - template - static void shuffle( V& vector ) { - RandomNumberGenerator rng; -#ifdef CATCH_CONFIG_CPP11_SHUFFLE - std::shuffle( vector.begin(), vector.end(), rng ); -#else - std::random_shuffle( vector.begin(), vector.end(), rng ); -#endif - } - }; - - inline std::vector sortTests( IConfig const& config, std::vector const& unsortedTestCases ) { - - std::vector sorted = unsortedTestCases; - - switch( config.runOrder() ) { - case RunTests::InLexicographicalOrder: - std::sort( sorted.begin(), sorted.end() ); - break; - case RunTests::InRandomOrder: - { - seedRng( config ); - RandomNumberGenerator::shuffle( sorted ); - } - break; - case RunTests::InDeclarationOrder: - // already in declaration order - break; - } - return sorted; - } - bool matchTest( TestCase const& testCase, TestSpec const& testSpec, IConfig const& config ) { - return testSpec.matches( testCase ) && ( config.allowThrows() || !testCase.throws() ); - } - - void enforceNoDuplicateTestCases( std::vector const& functions ) { - std::set seenFunctions; - for( std::vector::const_iterator it = functions.begin(), itEnd = functions.end(); - it != itEnd; - ++it ) { - std::pair::const_iterator, bool> prev = seenFunctions.insert( *it ); - if( !prev.second ) { - std::ostringstream ss; - - ss << Colour( Colour::Red ) - << "error: TEST_CASE( \"" << it->name << "\" ) already defined.\n" - << "\tFirst seen at " << prev.first->getTestCaseInfo().lineInfo << '\n' - << "\tRedefined at " << it->getTestCaseInfo().lineInfo << std::endl; - - throw std::runtime_error(ss.str()); - } - } - } - - std::vector filterTests( std::vector const& testCases, TestSpec const& testSpec, IConfig const& config ) { - std::vector filtered; - filtered.reserve( testCases.size() ); - for( std::vector::const_iterator it = testCases.begin(), itEnd = testCases.end(); - it != itEnd; - ++it ) - if( matchTest( *it, testSpec, config ) ) - filtered.push_back( *it ); - return filtered; - } - std::vector const& getAllTestCasesSorted( IConfig const& config ) { - return getRegistryHub().getTestCaseRegistry().getAllTestsSorted( config ); - } - - class TestRegistry : public ITestCaseRegistry { - public: - TestRegistry() - : m_currentSortOrder( RunTests::InDeclarationOrder ), - m_unnamedCount( 0 ) - {} - virtual ~TestRegistry(); - - virtual void registerTest( TestCase const& testCase ) { - std::string name = testCase.getTestCaseInfo().name; - if( name.empty() ) { - std::ostringstream oss; - oss << "Anonymous test case " << ++m_unnamedCount; - return registerTest( testCase.withName( oss.str() ) ); - } - m_functions.push_back( testCase ); - } - - virtual std::vector const& getAllTests() const { - return m_functions; - } - virtual std::vector const& getAllTestsSorted( IConfig const& config ) const { - if( m_sortedFunctions.empty() ) - enforceNoDuplicateTestCases( m_functions ); - - if( m_currentSortOrder != config.runOrder() || m_sortedFunctions.empty() ) { - m_sortedFunctions = sortTests( config, m_functions ); - m_currentSortOrder = config.runOrder(); - } - return m_sortedFunctions; - } - - private: - std::vector m_functions; - mutable RunTests::InWhatOrder m_currentSortOrder; - mutable std::vector m_sortedFunctions; - size_t m_unnamedCount; - std::ios_base::Init m_ostreamInit; // Forces cout/ cerr to be initialised - }; - - /////////////////////////////////////////////////////////////////////////// - - class FreeFunctionTestCase : public SharedImpl { - public: - - FreeFunctionTestCase( TestFunction fun ) : m_fun( fun ) {} - - virtual void invoke() const { - m_fun(); - } - - private: - virtual ~FreeFunctionTestCase(); - - TestFunction m_fun; - }; - - inline std::string extractClassName( std::string const& classOrQualifiedMethodName ) { - std::string className = classOrQualifiedMethodName; - if( startsWith( className, '&' ) ) - { - std::size_t lastColons = className.rfind( "::" ); - std::size_t penultimateColons = className.rfind( "::", lastColons-1 ); - if( penultimateColons == std::string::npos ) - penultimateColons = 1; - className = className.substr( penultimateColons, lastColons-penultimateColons ); - } - return className; - } - - void registerTestCase - ( ITestCase* testCase, - char const* classOrQualifiedMethodName, - NameAndDesc const& nameAndDesc, - SourceLineInfo const& lineInfo ) { - - getMutableRegistryHub().registerTest - ( makeTestCase - ( testCase, - extractClassName( classOrQualifiedMethodName ), - nameAndDesc.name, - nameAndDesc.description, - lineInfo ) ); - } - void registerTestCaseFunction - ( TestFunction function, - SourceLineInfo const& lineInfo, - NameAndDesc const& nameAndDesc ) { - registerTestCase( new FreeFunctionTestCase( function ), "", nameAndDesc, lineInfo ); - } - - /////////////////////////////////////////////////////////////////////////// - - AutoReg::AutoReg - ( TestFunction function, - SourceLineInfo const& lineInfo, - NameAndDesc const& nameAndDesc ) { - registerTestCaseFunction( function, lineInfo, nameAndDesc ); - } - - AutoReg::~AutoReg() {} - -} // end namespace Catch - -// #included from: catch_reporter_registry.hpp -#define TWOBLUECUBES_CATCH_REPORTER_REGISTRY_HPP_INCLUDED - -#include - -namespace Catch { - - class ReporterRegistry : public IReporterRegistry { - - public: - - virtual ~ReporterRegistry() CATCH_OVERRIDE {} - - virtual IStreamingReporter* create( std::string const& name, Ptr const& config ) const CATCH_OVERRIDE { - FactoryMap::const_iterator it = m_factories.find( name ); - if( it == m_factories.end() ) - return CATCH_NULL; - return it->second->create( ReporterConfig( config ) ); - } - - void registerReporter( std::string const& name, Ptr const& factory ) { - m_factories.insert( std::make_pair( name, factory ) ); - } - void registerListener( Ptr const& factory ) { - m_listeners.push_back( factory ); - } - - virtual FactoryMap const& getFactories() const CATCH_OVERRIDE { - return m_factories; - } - virtual Listeners const& getListeners() const CATCH_OVERRIDE { - return m_listeners; - } - - private: - FactoryMap m_factories; - Listeners m_listeners; - }; -} - -// #included from: catch_exception_translator_registry.hpp -#define TWOBLUECUBES_CATCH_EXCEPTION_TRANSLATOR_REGISTRY_HPP_INCLUDED - -#ifdef __OBJC__ -#import "Foundation/Foundation.h" -#endif - -namespace Catch { - - class ExceptionTranslatorRegistry : public IExceptionTranslatorRegistry { - public: - ~ExceptionTranslatorRegistry() { - deleteAll( m_translators ); - } - - virtual void registerTranslator( const IExceptionTranslator* translator ) { - m_translators.push_back( translator ); - } - - virtual std::string translateActiveException() const { - try { -#ifdef __OBJC__ - // In Objective-C try objective-c exceptions first - @try { - return tryTranslators(); - } - @catch (NSException *exception) { - return Catch::toString( [exception description] ); - } -#else - return tryTranslators(); -#endif - } - catch( TestFailureException& ) { - throw; - } - catch( std::exception& ex ) { - return ex.what(); - } - catch( std::string& msg ) { - return msg; - } - catch( const char* msg ) { - return msg; - } - catch(...) { - return "Unknown exception"; - } - } - - std::string tryTranslators() const { - if( m_translators.empty() ) - throw; - else - return m_translators[0]->translate( m_translators.begin()+1, m_translators.end() ); - } - - private: - std::vector m_translators; - }; -} - -// #included from: catch_tag_alias_registry.h -#define TWOBLUECUBES_CATCH_TAG_ALIAS_REGISTRY_H_INCLUDED - -#include - -namespace Catch { - - class TagAliasRegistry : public ITagAliasRegistry { - public: - virtual ~TagAliasRegistry(); - virtual Option find( std::string const& alias ) const; - virtual std::string expandAliases( std::string const& unexpandedTestSpec ) const; - void add( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ); - - private: - std::map m_registry; - }; - -} // end namespace Catch - -namespace Catch { - - namespace { - - class RegistryHub : public IRegistryHub, public IMutableRegistryHub { - - RegistryHub( RegistryHub const& ); - void operator=( RegistryHub const& ); - - public: // IRegistryHub - RegistryHub() { - } - virtual IReporterRegistry const& getReporterRegistry() const CATCH_OVERRIDE { - return m_reporterRegistry; - } - virtual ITestCaseRegistry const& getTestCaseRegistry() const CATCH_OVERRIDE { - return m_testCaseRegistry; - } - virtual IExceptionTranslatorRegistry& getExceptionTranslatorRegistry() CATCH_OVERRIDE { - return m_exceptionTranslatorRegistry; - } - virtual ITagAliasRegistry const& getTagAliasRegistry() const CATCH_OVERRIDE { - return m_tagAliasRegistry; - } - - public: // IMutableRegistryHub - virtual void registerReporter( std::string const& name, Ptr const& factory ) CATCH_OVERRIDE { - m_reporterRegistry.registerReporter( name, factory ); - } - virtual void registerListener( Ptr const& factory ) CATCH_OVERRIDE { - m_reporterRegistry.registerListener( factory ); - } - virtual void registerTest( TestCase const& testInfo ) CATCH_OVERRIDE { - m_testCaseRegistry.registerTest( testInfo ); - } - virtual void registerTranslator( const IExceptionTranslator* translator ) CATCH_OVERRIDE { - m_exceptionTranslatorRegistry.registerTranslator( translator ); - } - virtual void registerTagAlias( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) CATCH_OVERRIDE { - m_tagAliasRegistry.add( alias, tag, lineInfo ); - } - - private: - TestRegistry m_testCaseRegistry; - ReporterRegistry m_reporterRegistry; - ExceptionTranslatorRegistry m_exceptionTranslatorRegistry; - TagAliasRegistry m_tagAliasRegistry; - }; - - // Single, global, instance - inline RegistryHub*& getTheRegistryHub() { - static RegistryHub* theRegistryHub = CATCH_NULL; - if( !theRegistryHub ) - theRegistryHub = new RegistryHub(); - return theRegistryHub; - } - } - - IRegistryHub& getRegistryHub() { - return *getTheRegistryHub(); - } - IMutableRegistryHub& getMutableRegistryHub() { - return *getTheRegistryHub(); - } - void cleanUp() { - delete getTheRegistryHub(); - getTheRegistryHub() = CATCH_NULL; - cleanUpContext(); - } - std::string translateActiveException() { - return getRegistryHub().getExceptionTranslatorRegistry().translateActiveException(); - } - -} // end namespace Catch - -// #included from: catch_notimplemented_exception.hpp -#define TWOBLUECUBES_CATCH_NOTIMPLEMENTED_EXCEPTION_HPP_INCLUDED - -#include - -namespace Catch { - - NotImplementedException::NotImplementedException( SourceLineInfo const& lineInfo ) - : m_lineInfo( lineInfo ) { - std::ostringstream oss; - oss << lineInfo << ": function "; - oss << "not implemented"; - m_what = oss.str(); - } - - const char* NotImplementedException::what() const CATCH_NOEXCEPT { - return m_what.c_str(); - } - -} // end namespace Catch - -// #included from: catch_context_impl.hpp -#define TWOBLUECUBES_CATCH_CONTEXT_IMPL_HPP_INCLUDED - -// #included from: catch_stream.hpp -#define TWOBLUECUBES_CATCH_STREAM_HPP_INCLUDED - -#include -#include -#include - -namespace Catch { - - template - class StreamBufImpl : public StreamBufBase { - char data[bufferSize]; - WriterF m_writer; - - public: - StreamBufImpl() { - setp( data, data + sizeof(data) ); - } - - ~StreamBufImpl() CATCH_NOEXCEPT { - sync(); - } - - private: - int overflow( int c ) { - sync(); - - if( c != EOF ) { - if( pbase() == epptr() ) - m_writer( std::string( 1, static_cast( c ) ) ); - else - sputc( static_cast( c ) ); - } - return 0; - } - - int sync() { - if( pbase() != pptr() ) { - m_writer( std::string( pbase(), static_cast( pptr() - pbase() ) ) ); - setp( pbase(), epptr() ); - } - return 0; - } - }; - - /////////////////////////////////////////////////////////////////////////// - - FileStream::FileStream( std::string const& filename ) { - m_ofs.open( filename.c_str() ); - if( m_ofs.fail() ) { - std::ostringstream oss; - oss << "Unable to open file: '" << filename << '\''; - throw std::domain_error( oss.str() ); - } - } - - std::ostream& FileStream::stream() const { - return m_ofs; - } - - struct OutputDebugWriter { - - void operator()( std::string const&str ) { - writeToDebugConsole( str ); - } - }; - - DebugOutStream::DebugOutStream() - : m_streamBuf( new StreamBufImpl() ), - m_os( m_streamBuf.get() ) - {} - - std::ostream& DebugOutStream::stream() const { - return m_os; - } - - // Store the streambuf from cout up-front because - // cout may get redirected when running tests - CoutStream::CoutStream() - : m_os( Catch::cout().rdbuf() ) - {} - - std::ostream& CoutStream::stream() const { - return m_os; - } - -#ifndef CATCH_CONFIG_NOSTDOUT // If you #define this you must implement these functions - std::ostream& cout() { - return std::cout; - } - std::ostream& cerr() { - return std::cerr; - } - std::ostream& clog() { - return std::clog; - } -#endif -} - -namespace Catch { - - class Context : public IMutableContext { - - Context() : m_config( CATCH_NULL ), m_runner( CATCH_NULL ), m_resultCapture( CATCH_NULL ) {} - Context( Context const& ); - void operator=( Context const& ); - - public: - virtual ~Context() { - deleteAllValues( m_generatorsByTestName ); - } - - public: // IContext - virtual IResultCapture* getResultCapture() { - return m_resultCapture; - } - virtual IRunner* getRunner() { - return m_runner; - } - virtual size_t getGeneratorIndex( std::string const& fileInfo, size_t totalSize ) { - return getGeneratorsForCurrentTest() - .getGeneratorInfo( fileInfo, totalSize ) - .getCurrentIndex(); - } - virtual bool advanceGeneratorsForCurrentTest() { - IGeneratorsForTest* generators = findGeneratorsForCurrentTest(); - return generators && generators->moveNext(); - } - - virtual Ptr getConfig() const { - return m_config; - } - - public: // IMutableContext - virtual void setResultCapture( IResultCapture* resultCapture ) { - m_resultCapture = resultCapture; - } - virtual void setRunner( IRunner* runner ) { - m_runner = runner; - } - virtual void setConfig( Ptr const& config ) { - m_config = config; - } - - friend IMutableContext& getCurrentMutableContext(); - - private: - IGeneratorsForTest* findGeneratorsForCurrentTest() { - std::string testName = getResultCapture()->getCurrentTestName(); - - std::map::const_iterator it = - m_generatorsByTestName.find( testName ); - return it != m_generatorsByTestName.end() - ? it->second - : CATCH_NULL; - } - - IGeneratorsForTest& getGeneratorsForCurrentTest() { - IGeneratorsForTest* generators = findGeneratorsForCurrentTest(); - if( !generators ) { - std::string testName = getResultCapture()->getCurrentTestName(); - generators = createGeneratorsForTest(); - m_generatorsByTestName.insert( std::make_pair( testName, generators ) ); - } - return *generators; - } - - private: - Ptr m_config; - IRunner* m_runner; - IResultCapture* m_resultCapture; - std::map m_generatorsByTestName; - }; - - namespace { - Context* currentContext = CATCH_NULL; - } - IMutableContext& getCurrentMutableContext() { - if( !currentContext ) - currentContext = new Context(); - return *currentContext; - } - IContext& getCurrentContext() { - return getCurrentMutableContext(); - } - - void cleanUpContext() { - delete currentContext; - currentContext = CATCH_NULL; - } -} - -// #included from: catch_console_colour_impl.hpp -#define TWOBLUECUBES_CATCH_CONSOLE_COLOUR_IMPL_HPP_INCLUDED - -// #included from: catch_errno_guard.hpp -#define TWOBLUECUBES_CATCH_ERRNO_GUARD_HPP_INCLUDED - -#include - -namespace Catch { - - class ErrnoGuard { - public: - ErrnoGuard():m_oldErrno(errno){} - ~ErrnoGuard() { errno = m_oldErrno; } - private: - int m_oldErrno; - }; - -} - -namespace Catch { - namespace { - - struct IColourImpl { - virtual ~IColourImpl() {} - virtual void use( Colour::Code _colourCode ) = 0; - }; - - struct NoColourImpl : IColourImpl { - void use( Colour::Code ) {} - - static IColourImpl* instance() { - static NoColourImpl s_instance; - return &s_instance; - } - }; - - } // anon namespace -} // namespace Catch - -#if !defined( CATCH_CONFIG_COLOUR_NONE ) && !defined( CATCH_CONFIG_COLOUR_WINDOWS ) && !defined( CATCH_CONFIG_COLOUR_ANSI ) -# ifdef CATCH_PLATFORM_WINDOWS -# define CATCH_CONFIG_COLOUR_WINDOWS -# else -# define CATCH_CONFIG_COLOUR_ANSI -# endif -#endif - -#if defined ( CATCH_CONFIG_COLOUR_WINDOWS ) ///////////////////////////////////////// - -namespace Catch { -namespace { - - class Win32ColourImpl : public IColourImpl { - public: - Win32ColourImpl() : stdoutHandle( GetStdHandle(STD_OUTPUT_HANDLE) ) - { - CONSOLE_SCREEN_BUFFER_INFO csbiInfo; - GetConsoleScreenBufferInfo( stdoutHandle, &csbiInfo ); - originalForegroundAttributes = csbiInfo.wAttributes & ~( BACKGROUND_GREEN | BACKGROUND_RED | BACKGROUND_BLUE | BACKGROUND_INTENSITY ); - originalBackgroundAttributes = csbiInfo.wAttributes & ~( FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE | FOREGROUND_INTENSITY ); - } - - virtual void use( Colour::Code _colourCode ) { - switch( _colourCode ) { - case Colour::None: return setTextAttribute( originalForegroundAttributes ); - case Colour::White: return setTextAttribute( FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE ); - case Colour::Red: return setTextAttribute( FOREGROUND_RED ); - case Colour::Green: return setTextAttribute( FOREGROUND_GREEN ); - case Colour::Blue: return setTextAttribute( FOREGROUND_BLUE ); - case Colour::Cyan: return setTextAttribute( FOREGROUND_BLUE | FOREGROUND_GREEN ); - case Colour::Yellow: return setTextAttribute( FOREGROUND_RED | FOREGROUND_GREEN ); - case Colour::Grey: return setTextAttribute( 0 ); - - case Colour::LightGrey: return setTextAttribute( FOREGROUND_INTENSITY ); - case Colour::BrightRed: return setTextAttribute( FOREGROUND_INTENSITY | FOREGROUND_RED ); - case Colour::BrightGreen: return setTextAttribute( FOREGROUND_INTENSITY | FOREGROUND_GREEN ); - case Colour::BrightWhite: return setTextAttribute( FOREGROUND_INTENSITY | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE ); - - case Colour::Bright: throw std::logic_error( "not a colour" ); - } - } - - private: - void setTextAttribute( WORD _textAttribute ) { - SetConsoleTextAttribute( stdoutHandle, _textAttribute | originalBackgroundAttributes ); - } - HANDLE stdoutHandle; - WORD originalForegroundAttributes; - WORD originalBackgroundAttributes; - }; - - IColourImpl* platformColourInstance() { - static Win32ColourImpl s_instance; - - Ptr config = getCurrentContext().getConfig(); - UseColour::YesOrNo colourMode = config - ? config->useColour() - : UseColour::Auto; - if( colourMode == UseColour::Auto ) - colourMode = !isDebuggerActive() - ? UseColour::Yes - : UseColour::No; - return colourMode == UseColour::Yes - ? &s_instance - : NoColourImpl::instance(); - } - -} // end anon namespace -} // end namespace Catch - -#elif defined( CATCH_CONFIG_COLOUR_ANSI ) ////////////////////////////////////// - -#include - -namespace Catch { -namespace { - - // use POSIX/ ANSI console terminal codes - // Thanks to Adam Strzelecki for original contribution - // (http://github.com/nanoant) - // https://github.com/philsquared/Catch/pull/131 - class PosixColourImpl : public IColourImpl { - public: - virtual void use( Colour::Code _colourCode ) { - switch( _colourCode ) { - case Colour::None: - case Colour::White: return setColour( "[0m" ); - case Colour::Red: return setColour( "[0;31m" ); - case Colour::Green: return setColour( "[0;32m" ); - case Colour::Blue: return setColour( "[0;34m" ); - case Colour::Cyan: return setColour( "[0;36m" ); - case Colour::Yellow: return setColour( "[0;33m" ); - case Colour::Grey: return setColour( "[1;30m" ); - - case Colour::LightGrey: return setColour( "[0;37m" ); - case Colour::BrightRed: return setColour( "[1;31m" ); - case Colour::BrightGreen: return setColour( "[1;32m" ); - case Colour::BrightWhite: return setColour( "[1;37m" ); - - case Colour::Bright: throw std::logic_error( "not a colour" ); - } - } - static IColourImpl* instance() { - static PosixColourImpl s_instance; - return &s_instance; - } - - private: - void setColour( const char* _escapeCode ) { - Catch::cout() << '\033' << _escapeCode; - } - }; - - IColourImpl* platformColourInstance() { - ErrnoGuard guard; - Ptr config = getCurrentContext().getConfig(); - UseColour::YesOrNo colourMode = config - ? config->useColour() - : UseColour::Auto; - if( colourMode == UseColour::Auto ) - colourMode = (!isDebuggerActive() && isatty(STDOUT_FILENO) ) - ? UseColour::Yes - : UseColour::No; - return colourMode == UseColour::Yes - ? PosixColourImpl::instance() - : NoColourImpl::instance(); - } - -} // end anon namespace -} // end namespace Catch - -#else // not Windows or ANSI /////////////////////////////////////////////// - -namespace Catch { - - static IColourImpl* platformColourInstance() { return NoColourImpl::instance(); } - -} // end namespace Catch - -#endif // Windows/ ANSI/ None - -namespace Catch { - - Colour::Colour( Code _colourCode ) : m_moved( false ) { use( _colourCode ); } - Colour::Colour( Colour const& _other ) : m_moved( false ) { const_cast( _other ).m_moved = true; } - Colour::~Colour(){ if( !m_moved ) use( None ); } - - void Colour::use( Code _colourCode ) { - static IColourImpl* impl = platformColourInstance(); - impl->use( _colourCode ); - } - -} // end namespace Catch - -// #included from: catch_generators_impl.hpp -#define TWOBLUECUBES_CATCH_GENERATORS_IMPL_HPP_INCLUDED - -#include -#include -#include - -namespace Catch { - - struct GeneratorInfo : IGeneratorInfo { - - GeneratorInfo( std::size_t size ) - : m_size( size ), - m_currentIndex( 0 ) - {} - - bool moveNext() { - if( ++m_currentIndex == m_size ) { - m_currentIndex = 0; - return false; - } - return true; - } - - std::size_t getCurrentIndex() const { - return m_currentIndex; - } - - std::size_t m_size; - std::size_t m_currentIndex; - }; - - /////////////////////////////////////////////////////////////////////////// - - class GeneratorsForTest : public IGeneratorsForTest { - - public: - ~GeneratorsForTest() { - deleteAll( m_generatorsInOrder ); - } - - IGeneratorInfo& getGeneratorInfo( std::string const& fileInfo, std::size_t size ) { - std::map::const_iterator it = m_generatorsByName.find( fileInfo ); - if( it == m_generatorsByName.end() ) { - IGeneratorInfo* info = new GeneratorInfo( size ); - m_generatorsByName.insert( std::make_pair( fileInfo, info ) ); - m_generatorsInOrder.push_back( info ); - return *info; - } - return *it->second; - } - - bool moveNext() { - std::vector::const_iterator it = m_generatorsInOrder.begin(); - std::vector::const_iterator itEnd = m_generatorsInOrder.end(); - for(; it != itEnd; ++it ) { - if( (*it)->moveNext() ) - return true; - } - return false; - } - - private: - std::map m_generatorsByName; - std::vector m_generatorsInOrder; - }; - - IGeneratorsForTest* createGeneratorsForTest() - { - return new GeneratorsForTest(); - } - -} // end namespace Catch - -// #included from: catch_assertionresult.hpp -#define TWOBLUECUBES_CATCH_ASSERTIONRESULT_HPP_INCLUDED - -namespace Catch { - - AssertionInfo::AssertionInfo():macroName(""), capturedExpression(""), resultDisposition(ResultDisposition::Normal), secondArg(""){} - - AssertionInfo::AssertionInfo( char const * _macroName, - SourceLineInfo const& _lineInfo, - char const * _capturedExpression, - ResultDisposition::Flags _resultDisposition, - char const * _secondArg) - : macroName( _macroName ), - lineInfo( _lineInfo ), - capturedExpression( _capturedExpression ), - resultDisposition( _resultDisposition ), - secondArg( _secondArg ) - {} - - AssertionResult::AssertionResult() {} - - AssertionResult::AssertionResult( AssertionInfo const& info, AssertionResultData const& data ) - : m_info( info ), - m_resultData( data ) - {} - - AssertionResult::~AssertionResult() {} - - // Result was a success - bool AssertionResult::succeeded() const { - return Catch::isOk( m_resultData.resultType ); - } - - // Result was a success, or failure is suppressed - bool AssertionResult::isOk() const { - return Catch::isOk( m_resultData.resultType ) || shouldSuppressFailure( m_info.resultDisposition ); - } - - ResultWas::OfType AssertionResult::getResultType() const { - return m_resultData.resultType; - } - - bool AssertionResult::hasExpression() const { - return m_info.capturedExpression[0] != 0; - } - - bool AssertionResult::hasMessage() const { - return !m_resultData.message.empty(); - } - - std::string capturedExpressionWithSecondArgument( char const * capturedExpression, char const * secondArg ) { - return (secondArg[0] == 0 || secondArg[0] == '"' && secondArg[1] == '"') - ? capturedExpression - : std::string(capturedExpression) + ", " + secondArg; - } - - std::string AssertionResult::getExpression() const { - if( isFalseTest( m_info.resultDisposition ) ) - return "!(" + capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg) + ")"; - else - return capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg); - } - std::string AssertionResult::getExpressionInMacro() const { - if( m_info.macroName[0] == 0 ) - return capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg); - else - return std::string(m_info.macroName) + "( " + capturedExpressionWithSecondArgument(m_info.capturedExpression, m_info.secondArg) + " )"; - } - - bool AssertionResult::hasExpandedExpression() const { - return hasExpression() && getExpandedExpression() != getExpression(); - } - - std::string AssertionResult::getExpandedExpression() const { - return m_resultData.reconstructExpression(); - } - - std::string AssertionResult::getMessage() const { - return m_resultData.message; - } - SourceLineInfo AssertionResult::getSourceInfo() const { - return m_info.lineInfo; - } - - std::string AssertionResult::getTestMacroName() const { - return m_info.macroName; - } - - void AssertionResult::discardDecomposedExpression() const { - m_resultData.decomposedExpression = CATCH_NULL; - } - - void AssertionResult::expandDecomposedExpression() const { - m_resultData.reconstructExpression(); - } - -} // end namespace Catch - -// #included from: catch_test_case_info.hpp -#define TWOBLUECUBES_CATCH_TEST_CASE_INFO_HPP_INCLUDED - -#include - -namespace Catch { - - inline TestCaseInfo::SpecialProperties parseSpecialTag( std::string const& tag ) { - if( startsWith( tag, '.' ) || - tag == "hide" || - tag == "!hide" ) - return TestCaseInfo::IsHidden; - else if( tag == "!throws" ) - return TestCaseInfo::Throws; - else if( tag == "!shouldfail" ) - return TestCaseInfo::ShouldFail; - else if( tag == "!mayfail" ) - return TestCaseInfo::MayFail; - else if( tag == "!nonportable" ) - return TestCaseInfo::NonPortable; - else - return TestCaseInfo::None; - } - inline bool isReservedTag( std::string const& tag ) { - return parseSpecialTag( tag ) == TestCaseInfo::None && tag.size() > 0 && !std::isalnum( tag[0] ); - } - inline void enforceNotReservedTag( std::string const& tag, SourceLineInfo const& _lineInfo ) { - if( isReservedTag( tag ) ) { - std::ostringstream ss; - ss << Colour(Colour::Red) - << "Tag name [" << tag << "] not allowed.\n" - << "Tag names starting with non alpha-numeric characters are reserved\n" - << Colour(Colour::FileName) - << _lineInfo << '\n'; - throw std::runtime_error(ss.str()); - } - } - - TestCase makeTestCase( ITestCase* _testCase, - std::string const& _className, - std::string const& _name, - std::string const& _descOrTags, - SourceLineInfo const& _lineInfo ) - { - bool isHidden( startsWith( _name, "./" ) ); // Legacy support - - // Parse out tags - std::set tags; - std::string desc, tag; - bool inTag = false; - for( std::size_t i = 0; i < _descOrTags.size(); ++i ) { - char c = _descOrTags[i]; - if( !inTag ) { - if( c == '[' ) - inTag = true; - else - desc += c; - } - else { - if( c == ']' ) { - TestCaseInfo::SpecialProperties prop = parseSpecialTag( tag ); - if( prop == TestCaseInfo::IsHidden ) - isHidden = true; - else if( prop == TestCaseInfo::None ) - enforceNotReservedTag( tag, _lineInfo ); - - tags.insert( tag ); - tag.clear(); - inTag = false; - } - else - tag += c; - } - } - if( isHidden ) { - tags.insert( "hide" ); - tags.insert( "." ); - } - - TestCaseInfo info( _name, _className, desc, tags, _lineInfo ); - return TestCase( _testCase, info ); - } - - void setTags( TestCaseInfo& testCaseInfo, std::set const& tags ) - { - testCaseInfo.tags = tags; - testCaseInfo.lcaseTags.clear(); - - std::ostringstream oss; - for( std::set::const_iterator it = tags.begin(), itEnd = tags.end(); it != itEnd; ++it ) { - oss << '[' << *it << ']'; - std::string lcaseTag = toLower( *it ); - testCaseInfo.properties = static_cast( testCaseInfo.properties | parseSpecialTag( lcaseTag ) ); - testCaseInfo.lcaseTags.insert( lcaseTag ); - } - testCaseInfo.tagsAsString = oss.str(); - } - - TestCaseInfo::TestCaseInfo( std::string const& _name, - std::string const& _className, - std::string const& _description, - std::set const& _tags, - SourceLineInfo const& _lineInfo ) - : name( _name ), - className( _className ), - description( _description ), - lineInfo( _lineInfo ), - properties( None ) - { - setTags( *this, _tags ); - } - - TestCaseInfo::TestCaseInfo( TestCaseInfo const& other ) - : name( other.name ), - className( other.className ), - description( other.description ), - tags( other.tags ), - lcaseTags( other.lcaseTags ), - tagsAsString( other.tagsAsString ), - lineInfo( other.lineInfo ), - properties( other.properties ) - {} - - bool TestCaseInfo::isHidden() const { - return ( properties & IsHidden ) != 0; - } - bool TestCaseInfo::throws() const { - return ( properties & Throws ) != 0; - } - bool TestCaseInfo::okToFail() const { - return ( properties & (ShouldFail | MayFail ) ) != 0; - } - bool TestCaseInfo::expectedToFail() const { - return ( properties & (ShouldFail ) ) != 0; - } - - TestCase::TestCase( ITestCase* testCase, TestCaseInfo const& info ) : TestCaseInfo( info ), test( testCase ) {} - - TestCase::TestCase( TestCase const& other ) - : TestCaseInfo( other ), - test( other.test ) - {} - - TestCase TestCase::withName( std::string const& _newName ) const { - TestCase other( *this ); - other.name = _newName; - return other; - } - - void TestCase::swap( TestCase& other ) { - test.swap( other.test ); - name.swap( other.name ); - className.swap( other.className ); - description.swap( other.description ); - tags.swap( other.tags ); - lcaseTags.swap( other.lcaseTags ); - tagsAsString.swap( other.tagsAsString ); - std::swap( TestCaseInfo::properties, static_cast( other ).properties ); - std::swap( lineInfo, other.lineInfo ); - } - - void TestCase::invoke() const { - test->invoke(); - } - - bool TestCase::operator == ( TestCase const& other ) const { - return test.get() == other.test.get() && - name == other.name && - className == other.className; - } - - bool TestCase::operator < ( TestCase const& other ) const { - return name < other.name; - } - TestCase& TestCase::operator = ( TestCase const& other ) { - TestCase temp( other ); - swap( temp ); - return *this; - } - - TestCaseInfo const& TestCase::getTestCaseInfo() const - { - return *this; - } - -} // end namespace Catch - -// #included from: catch_version.hpp -#define TWOBLUECUBES_CATCH_VERSION_HPP_INCLUDED - -namespace Catch { - - Version::Version - ( unsigned int _majorVersion, - unsigned int _minorVersion, - unsigned int _patchNumber, - char const * const _branchName, - unsigned int _buildNumber ) - : majorVersion( _majorVersion ), - minorVersion( _minorVersion ), - patchNumber( _patchNumber ), - branchName( _branchName ), - buildNumber( _buildNumber ) - {} - - std::ostream& operator << ( std::ostream& os, Version const& version ) { - os << version.majorVersion << '.' - << version.minorVersion << '.' - << version.patchNumber; - // branchName is never null -> 0th char is \0 if it is empty - if (version.branchName[0]) { - os << '-' << version.branchName - << '.' << version.buildNumber; - } - return os; - } - - inline Version libraryVersion() { - static Version version( 1, 12, 2, "", 0 ); - return version; - } - -} - -// #included from: catch_message.hpp -#define TWOBLUECUBES_CATCH_MESSAGE_HPP_INCLUDED - -namespace Catch { - - MessageInfo::MessageInfo( std::string const& _macroName, - SourceLineInfo const& _lineInfo, - ResultWas::OfType _type ) - : macroName( _macroName ), - lineInfo( _lineInfo ), - type( _type ), - sequence( ++globalCount ) - {} - - // This may need protecting if threading support is added - unsigned int MessageInfo::globalCount = 0; - - //////////////////////////////////////////////////////////////////////////// - - ScopedMessage::ScopedMessage( MessageBuilder const& builder ) - : m_info( builder.m_info ) - { - m_info.message = builder.m_stream.str(); - getResultCapture().pushScopedMessage( m_info ); - } - ScopedMessage::ScopedMessage( ScopedMessage const& other ) - : m_info( other.m_info ) - {} - -#if defined(_MSC_VER) -#pragma warning(push) -#pragma warning(disable:4996) // std::uncaught_exception is deprecated in C++17 -#endif - ScopedMessage::~ScopedMessage() { - if ( !std::uncaught_exception() ){ - getResultCapture().popScopedMessage(m_info); - } - } -#if defined(_MSC_VER) -#pragma warning(pop) -#endif - -} // end namespace Catch - -// #included from: catch_legacy_reporter_adapter.hpp -#define TWOBLUECUBES_CATCH_LEGACY_REPORTER_ADAPTER_HPP_INCLUDED - -// #included from: catch_legacy_reporter_adapter.h -#define TWOBLUECUBES_CATCH_LEGACY_REPORTER_ADAPTER_H_INCLUDED - -namespace Catch -{ - // Deprecated - struct IReporter : IShared { - virtual ~IReporter(); - - virtual bool shouldRedirectStdout() const = 0; - - virtual void StartTesting() = 0; - virtual void EndTesting( Totals const& totals ) = 0; - virtual void StartGroup( std::string const& groupName ) = 0; - virtual void EndGroup( std::string const& groupName, Totals const& totals ) = 0; - virtual void StartTestCase( TestCaseInfo const& testInfo ) = 0; - virtual void EndTestCase( TestCaseInfo const& testInfo, Totals const& totals, std::string const& stdOut, std::string const& stdErr ) = 0; - virtual void StartSection( std::string const& sectionName, std::string const& description ) = 0; - virtual void EndSection( std::string const& sectionName, Counts const& assertions ) = 0; - virtual void NoAssertionsInSection( std::string const& sectionName ) = 0; - virtual void NoAssertionsInTestCase( std::string const& testName ) = 0; - virtual void Aborted() = 0; - virtual void Result( AssertionResult const& result ) = 0; - }; - - class LegacyReporterAdapter : public SharedImpl - { - public: - LegacyReporterAdapter( Ptr const& legacyReporter ); - virtual ~LegacyReporterAdapter(); - - virtual ReporterPreferences getPreferences() const; - virtual void noMatchingTestCases( std::string const& ); - virtual void testRunStarting( TestRunInfo const& ); - virtual void testGroupStarting( GroupInfo const& groupInfo ); - virtual void testCaseStarting( TestCaseInfo const& testInfo ); - virtual void sectionStarting( SectionInfo const& sectionInfo ); - virtual void assertionStarting( AssertionInfo const& ); - virtual bool assertionEnded( AssertionStats const& assertionStats ); - virtual void sectionEnded( SectionStats const& sectionStats ); - virtual void testCaseEnded( TestCaseStats const& testCaseStats ); - virtual void testGroupEnded( TestGroupStats const& testGroupStats ); - virtual void testRunEnded( TestRunStats const& testRunStats ); - virtual void skipTest( TestCaseInfo const& ); - - private: - Ptr m_legacyReporter; - }; -} - -namespace Catch -{ - LegacyReporterAdapter::LegacyReporterAdapter( Ptr const& legacyReporter ) - : m_legacyReporter( legacyReporter ) - {} - LegacyReporterAdapter::~LegacyReporterAdapter() {} - - ReporterPreferences LegacyReporterAdapter::getPreferences() const { - ReporterPreferences prefs; - prefs.shouldRedirectStdOut = m_legacyReporter->shouldRedirectStdout(); - return prefs; - } - - void LegacyReporterAdapter::noMatchingTestCases( std::string const& ) {} - void LegacyReporterAdapter::testRunStarting( TestRunInfo const& ) { - m_legacyReporter->StartTesting(); - } - void LegacyReporterAdapter::testGroupStarting( GroupInfo const& groupInfo ) { - m_legacyReporter->StartGroup( groupInfo.name ); - } - void LegacyReporterAdapter::testCaseStarting( TestCaseInfo const& testInfo ) { - m_legacyReporter->StartTestCase( testInfo ); - } - void LegacyReporterAdapter::sectionStarting( SectionInfo const& sectionInfo ) { - m_legacyReporter->StartSection( sectionInfo.name, sectionInfo.description ); - } - void LegacyReporterAdapter::assertionStarting( AssertionInfo const& ) { - // Not on legacy interface - } - - bool LegacyReporterAdapter::assertionEnded( AssertionStats const& assertionStats ) { - if( assertionStats.assertionResult.getResultType() != ResultWas::Ok ) { - for( std::vector::const_iterator it = assertionStats.infoMessages.begin(), itEnd = assertionStats.infoMessages.end(); - it != itEnd; - ++it ) { - if( it->type == ResultWas::Info ) { - ResultBuilder rb( it->macroName.c_str(), it->lineInfo, "", ResultDisposition::Normal ); - rb << it->message; - rb.setResultType( ResultWas::Info ); - AssertionResult result = rb.build(); - m_legacyReporter->Result( result ); - } - } - } - m_legacyReporter->Result( assertionStats.assertionResult ); - return true; - } - void LegacyReporterAdapter::sectionEnded( SectionStats const& sectionStats ) { - if( sectionStats.missingAssertions ) - m_legacyReporter->NoAssertionsInSection( sectionStats.sectionInfo.name ); - m_legacyReporter->EndSection( sectionStats.sectionInfo.name, sectionStats.assertions ); - } - void LegacyReporterAdapter::testCaseEnded( TestCaseStats const& testCaseStats ) { - m_legacyReporter->EndTestCase - ( testCaseStats.testInfo, - testCaseStats.totals, - testCaseStats.stdOut, - testCaseStats.stdErr ); - } - void LegacyReporterAdapter::testGroupEnded( TestGroupStats const& testGroupStats ) { - if( testGroupStats.aborting ) - m_legacyReporter->Aborted(); - m_legacyReporter->EndGroup( testGroupStats.groupInfo.name, testGroupStats.totals ); - } - void LegacyReporterAdapter::testRunEnded( TestRunStats const& testRunStats ) { - m_legacyReporter->EndTesting( testRunStats.totals ); - } - void LegacyReporterAdapter::skipTest( TestCaseInfo const& ) { - } -} - -// #included from: catch_timer.hpp - -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wc++11-long-long" -#endif - -#ifdef CATCH_PLATFORM_WINDOWS - -#else - -#include - -#endif - -namespace Catch { - - namespace { -#ifdef CATCH_PLATFORM_WINDOWS - UInt64 getCurrentTicks() { - static UInt64 hz=0, hzo=0; - if (!hz) { - QueryPerformanceFrequency( reinterpret_cast( &hz ) ); - QueryPerformanceCounter( reinterpret_cast( &hzo ) ); - } - UInt64 t; - QueryPerformanceCounter( reinterpret_cast( &t ) ); - return ((t-hzo)*1000000)/hz; - } -#else - UInt64 getCurrentTicks() { - timeval t; - gettimeofday(&t,CATCH_NULL); - return static_cast( t.tv_sec ) * 1000000ull + static_cast( t.tv_usec ); - } -#endif - } - - void Timer::start() { - m_ticks = getCurrentTicks(); - } - unsigned int Timer::getElapsedMicroseconds() const { - return static_cast(getCurrentTicks() - m_ticks); - } - unsigned int Timer::getElapsedMilliseconds() const { - return static_cast(getElapsedMicroseconds()/1000); - } - double Timer::getElapsedSeconds() const { - return getElapsedMicroseconds()/1000000.0; - } - -} // namespace Catch - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif -// #included from: catch_common.hpp -#define TWOBLUECUBES_CATCH_COMMON_HPP_INCLUDED - -#include -#include - -namespace Catch { - - bool startsWith( std::string const& s, std::string const& prefix ) { - return s.size() >= prefix.size() && std::equal(prefix.begin(), prefix.end(), s.begin()); - } - bool startsWith( std::string const& s, char prefix ) { - return !s.empty() && s[0] == prefix; - } - bool endsWith( std::string const& s, std::string const& suffix ) { - return s.size() >= suffix.size() && std::equal(suffix.rbegin(), suffix.rend(), s.rbegin()); - } - bool endsWith( std::string const& s, char suffix ) { - return !s.empty() && s[s.size()-1] == suffix; - } - bool contains( std::string const& s, std::string const& infix ) { - return s.find( infix ) != std::string::npos; - } - char toLowerCh(char c) { - return static_cast( std::tolower( c ) ); - } - void toLowerInPlace( std::string& s ) { - std::transform( s.begin(), s.end(), s.begin(), toLowerCh ); - } - std::string toLower( std::string const& s ) { - std::string lc = s; - toLowerInPlace( lc ); - return lc; - } - std::string trim( std::string const& str ) { - static char const* whitespaceChars = "\n\r\t "; - std::string::size_type start = str.find_first_not_of( whitespaceChars ); - std::string::size_type end = str.find_last_not_of( whitespaceChars ); - - return start != std::string::npos ? str.substr( start, 1+end-start ) : std::string(); - } - - bool replaceInPlace( std::string& str, std::string const& replaceThis, std::string const& withThis ) { - bool replaced = false; - std::size_t i = str.find( replaceThis ); - while( i != std::string::npos ) { - replaced = true; - str = str.substr( 0, i ) + withThis + str.substr( i+replaceThis.size() ); - if( i < str.size()-withThis.size() ) - i = str.find( replaceThis, i+withThis.size() ); - else - i = std::string::npos; - } - return replaced; - } - - pluralise::pluralise( std::size_t count, std::string const& label ) - : m_count( count ), - m_label( label ) - {} - - std::ostream& operator << ( std::ostream& os, pluralise const& pluraliser ) { - os << pluraliser.m_count << ' ' << pluraliser.m_label; - if( pluraliser.m_count != 1 ) - os << 's'; - return os; - } - - SourceLineInfo::SourceLineInfo() : file(""), line( 0 ){} - SourceLineInfo::SourceLineInfo( char const* _file, std::size_t _line ) - : file( _file ), - line( _line ) - {} - bool SourceLineInfo::empty() const { - return file[0] == '\0'; - } - bool SourceLineInfo::operator == ( SourceLineInfo const& other ) const { - return line == other.line && (file == other.file || std::strcmp(file, other.file) == 0); - } - bool SourceLineInfo::operator < ( SourceLineInfo const& other ) const { - return line < other.line || ( line == other.line && (std::strcmp(file, other.file) < 0)); - } - - void seedRng( IConfig const& config ) { - if( config.rngSeed() != 0 ) - std::srand( config.rngSeed() ); - } - unsigned int rngSeed() { - return getCurrentContext().getConfig()->rngSeed(); - } - - std::ostream& operator << ( std::ostream& os, SourceLineInfo const& info ) { -#ifndef __GNUG__ - os << info.file << '(' << info.line << ')'; -#else - os << info.file << ':' << info.line; -#endif - return os; - } - - void throwLogicError( std::string const& message, SourceLineInfo const& locationInfo ) { - std::ostringstream oss; - oss << locationInfo << ": Internal Catch error: '" << message << '\''; - if( alwaysTrue() ) - throw std::logic_error( oss.str() ); - } -} - -// #included from: catch_section.hpp -#define TWOBLUECUBES_CATCH_SECTION_HPP_INCLUDED - -namespace Catch { - - SectionInfo::SectionInfo - ( SourceLineInfo const& _lineInfo, - std::string const& _name, - std::string const& _description ) - : name( _name ), - description( _description ), - lineInfo( _lineInfo ) - {} - - Section::Section( SectionInfo const& info ) - : m_info( info ), - m_sectionIncluded( getResultCapture().sectionStarted( m_info, m_assertions ) ) - { - m_timer.start(); - } - -#if defined(_MSC_VER) -#pragma warning(push) -#pragma warning(disable:4996) // std::uncaught_exception is deprecated in C++17 -#endif - Section::~Section() { - if( m_sectionIncluded ) { - SectionEndInfo endInfo( m_info, m_assertions, m_timer.getElapsedSeconds() ); - if( std::uncaught_exception() ) - getResultCapture().sectionEndedEarly( endInfo ); - else - getResultCapture().sectionEnded( endInfo ); - } - } -#if defined(_MSC_VER) -#pragma warning(pop) -#endif - - // This indicates whether the section should be executed or not - Section::operator bool() const { - return m_sectionIncluded; - } - -} // end namespace Catch - -// #included from: catch_debugger.hpp -#define TWOBLUECUBES_CATCH_DEBUGGER_HPP_INCLUDED - -#ifdef CATCH_PLATFORM_MAC - - #include - #include - #include - #include - #include - - namespace Catch{ - - // The following function is taken directly from the following technical note: - // http://developer.apple.com/library/mac/#qa/qa2004/qa1361.html - - // Returns true if the current process is being debugged (either - // running under the debugger or has a debugger attached post facto). - bool isDebuggerActive(){ - - int mib[4]; - struct kinfo_proc info; - size_t size; - - // Initialize the flags so that, if sysctl fails for some bizarre - // reason, we get a predictable result. - - info.kp_proc.p_flag = 0; - - // Initialize mib, which tells sysctl the info we want, in this case - // we're looking for information about a specific process ID. - - mib[0] = CTL_KERN; - mib[1] = KERN_PROC; - mib[2] = KERN_PROC_PID; - mib[3] = getpid(); - - // Call sysctl. - - size = sizeof(info); - if( sysctl(mib, sizeof(mib) / sizeof(*mib), &info, &size, CATCH_NULL, 0) != 0 ) { - Catch::cerr() << "\n** Call to sysctl failed - unable to determine if debugger is active **\n" << std::endl; - return false; - } - - // We're being debugged if the P_TRACED flag is set. - - return ( (info.kp_proc.p_flag & P_TRACED) != 0 ); - } - } // namespace Catch - -#elif defined(CATCH_PLATFORM_LINUX) - #include - #include - - namespace Catch{ - // The standard POSIX way of detecting a debugger is to attempt to - // ptrace() the process, but this needs to be done from a child and not - // this process itself to still allow attaching to this process later - // if wanted, so is rather heavy. Under Linux we have the PID of the - // "debugger" (which doesn't need to be gdb, of course, it could also - // be strace, for example) in /proc/$PID/status, so just get it from - // there instead. - bool isDebuggerActive(){ - // Libstdc++ has a bug, where std::ifstream sets errno to 0 - // This way our users can properly assert over errno values - ErrnoGuard guard; - std::ifstream in("/proc/self/status"); - for( std::string line; std::getline(in, line); ) { - static const int PREFIX_LEN = 11; - if( line.compare(0, PREFIX_LEN, "TracerPid:\t") == 0 ) { - // We're traced if the PID is not 0 and no other PID starts - // with 0 digit, so it's enough to check for just a single - // character. - return line.length() > PREFIX_LEN && line[PREFIX_LEN] != '0'; - } - } - - return false; - } - } // namespace Catch -#elif defined(_MSC_VER) - extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent(); - namespace Catch { - bool isDebuggerActive() { - return IsDebuggerPresent() != 0; - } - } -#elif defined(__MINGW32__) - extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent(); - namespace Catch { - bool isDebuggerActive() { - return IsDebuggerPresent() != 0; - } - } -#else - namespace Catch { - inline bool isDebuggerActive() { return false; } - } -#endif // Platform - -#ifdef CATCH_PLATFORM_WINDOWS - - namespace Catch { - void writeToDebugConsole( std::string const& text ) { - ::OutputDebugStringA( text.c_str() ); - } - } -#else - namespace Catch { - void writeToDebugConsole( std::string const& text ) { - // !TBD: Need a version for Mac/ XCode and other IDEs - Catch::cout() << text; - } - } -#endif // Platform - -// #included from: catch_tostring.hpp -#define TWOBLUECUBES_CATCH_TOSTRING_HPP_INCLUDED - -namespace Catch { - -namespace Detail { - - const std::string unprintableString = "{?}"; - - namespace { - const int hexThreshold = 255; - - struct Endianness { - enum Arch { Big, Little }; - - static Arch which() { - union _{ - int asInt; - char asChar[sizeof (int)]; - } u; - - u.asInt = 1; - return ( u.asChar[sizeof(int)-1] == 1 ) ? Big : Little; - } - }; - } - - std::string rawMemoryToString( const void *object, std::size_t size ) - { - // Reverse order for little endian architectures - int i = 0, end = static_cast( size ), inc = 1; - if( Endianness::which() == Endianness::Little ) { - i = end-1; - end = inc = -1; - } - - unsigned char const *bytes = static_cast(object); - std::ostringstream os; - os << "0x" << std::setfill('0') << std::hex; - for( ; i != end; i += inc ) - os << std::setw(2) << static_cast(bytes[i]); - return os.str(); - } -} - -std::string toString( std::string const& value ) { - std::string s = value; - if( getCurrentContext().getConfig()->showInvisibles() ) { - for(size_t i = 0; i < s.size(); ++i ) { - std::string subs; - switch( s[i] ) { - case '\n': subs = "\\n"; break; - case '\t': subs = "\\t"; break; - default: break; - } - if( !subs.empty() ) { - s = s.substr( 0, i ) + subs + s.substr( i+1 ); - ++i; - } - } - } - return '"' + s + '"'; -} -std::string toString( std::wstring const& value ) { - - std::string s; - s.reserve( value.size() ); - for(size_t i = 0; i < value.size(); ++i ) - s += value[i] <= 0xff ? static_cast( value[i] ) : '?'; - return Catch::toString( s ); -} - -std::string toString( const char* const value ) { - return value ? Catch::toString( std::string( value ) ) : std::string( "{null string}" ); -} - -std::string toString( char* const value ) { - return Catch::toString( static_cast( value ) ); -} - -std::string toString( const wchar_t* const value ) -{ - return value ? Catch::toString( std::wstring(value) ) : std::string( "{null string}" ); -} - -std::string toString( wchar_t* const value ) -{ - return Catch::toString( static_cast( value ) ); -} - -std::string toString( int value ) { - std::ostringstream oss; - oss << value; - if( value > Detail::hexThreshold ) - oss << " (0x" << std::hex << value << ')'; - return oss.str(); -} - -std::string toString( unsigned long value ) { - std::ostringstream oss; - oss << value; - if( value > Detail::hexThreshold ) - oss << " (0x" << std::hex << value << ')'; - return oss.str(); -} - -std::string toString( unsigned int value ) { - return Catch::toString( static_cast( value ) ); -} - -template -std::string fpToString( T value, int precision ) { - std::ostringstream oss; - oss << std::setprecision( precision ) - << std::fixed - << value; - std::string d = oss.str(); - std::size_t i = d.find_last_not_of( '0' ); - if( i != std::string::npos && i != d.size()-1 ) { - if( d[i] == '.' ) - i++; - d = d.substr( 0, i+1 ); - } - return d; -} - -std::string toString( const double value ) { - return fpToString( value, 10 ); -} -std::string toString( const float value ) { - return fpToString( value, 5 ) + 'f'; -} - -std::string toString( bool value ) { - return value ? "true" : "false"; -} - -std::string toString( char value ) { - if ( value == '\r' ) - return "'\\r'"; - if ( value == '\f' ) - return "'\\f'"; - if ( value == '\n' ) - return "'\\n'"; - if ( value == '\t' ) - return "'\\t'"; - if ( '\0' <= value && value < ' ' ) - return toString( static_cast( value ) ); - char chstr[] = "' '"; - chstr[1] = value; - return chstr; -} - -std::string toString( signed char value ) { - return toString( static_cast( value ) ); -} - -std::string toString( unsigned char value ) { - return toString( static_cast( value ) ); -} - -#ifdef CATCH_CONFIG_CPP11_LONG_LONG -std::string toString( long long value ) { - std::ostringstream oss; - oss << value; - if( value > Detail::hexThreshold ) - oss << " (0x" << std::hex << value << ')'; - return oss.str(); -} -std::string toString( unsigned long long value ) { - std::ostringstream oss; - oss << value; - if( value > Detail::hexThreshold ) - oss << " (0x" << std::hex << value << ')'; - return oss.str(); -} -#endif - -#ifdef CATCH_CONFIG_CPP11_NULLPTR -std::string toString( std::nullptr_t ) { - return "nullptr"; -} -#endif - -#ifdef __OBJC__ - std::string toString( NSString const * const& nsstring ) { - if( !nsstring ) - return "nil"; - return "@" + toString([nsstring UTF8String]); - } - std::string toString( NSString * CATCH_ARC_STRONG & nsstring ) { - if( !nsstring ) - return "nil"; - return "@" + toString([nsstring UTF8String]); - } - std::string toString( NSObject* const& nsObject ) { - return toString( [nsObject description] ); - } -#endif - -} // end namespace Catch - -// #included from: catch_result_builder.hpp -#define TWOBLUECUBES_CATCH_RESULT_BUILDER_HPP_INCLUDED - -#include - -namespace Catch { - - ResultBuilder::ResultBuilder( char const* macroName, - SourceLineInfo const& lineInfo, - char const* capturedExpression, - ResultDisposition::Flags resultDisposition, - char const* secondArg ) - : m_assertionInfo( macroName, lineInfo, capturedExpression, resultDisposition, secondArg ), - m_shouldDebugBreak( false ), - m_shouldThrow( false ), - m_guardException( false ), - m_usedStream( false ) - {} - - ResultBuilder::~ResultBuilder() { -#if defined(CATCH_CONFIG_FAST_COMPILE) - if ( m_guardException ) { - stream().oss << "Exception translation was disabled by CATCH_CONFIG_FAST_COMPILE"; - captureResult( ResultWas::ThrewException ); - getCurrentContext().getResultCapture()->exceptionEarlyReported(); - } -#endif - } - - ResultBuilder& ResultBuilder::setResultType( ResultWas::OfType result ) { - m_data.resultType = result; - return *this; - } - ResultBuilder& ResultBuilder::setResultType( bool result ) { - m_data.resultType = result ? ResultWas::Ok : ResultWas::ExpressionFailed; - return *this; - } - - void ResultBuilder::endExpression( DecomposedExpression const& expr ) { - // Flip bool results if FalseTest flag is set - if( isFalseTest( m_assertionInfo.resultDisposition ) ) { - m_data.negate( expr.isBinaryExpression() ); - } - - getResultCapture().assertionRun(); - - if(getCurrentContext().getConfig()->includeSuccessfulResults() || m_data.resultType != ResultWas::Ok) - { - AssertionResult result = build( expr ); - handleResult( result ); - } - else - getResultCapture().assertionPassed(); - } - - void ResultBuilder::useActiveException( ResultDisposition::Flags resultDisposition ) { - m_assertionInfo.resultDisposition = resultDisposition; - stream().oss << Catch::translateActiveException(); - captureResult( ResultWas::ThrewException ); - } - - void ResultBuilder::captureResult( ResultWas::OfType resultType ) { - setResultType( resultType ); - captureExpression(); - } - - void ResultBuilder::captureExpectedException( std::string const& expectedMessage ) { - if( expectedMessage.empty() ) - captureExpectedException( Matchers::Impl::MatchAllOf() ); - else - captureExpectedException( Matchers::Equals( expectedMessage ) ); - } - - void ResultBuilder::captureExpectedException( Matchers::Impl::MatcherBase const& matcher ) { - - assert( !isFalseTest( m_assertionInfo.resultDisposition ) ); - AssertionResultData data = m_data; - data.resultType = ResultWas::Ok; - data.reconstructedExpression = capturedExpressionWithSecondArgument(m_assertionInfo.capturedExpression, m_assertionInfo.secondArg); - - std::string actualMessage = Catch::translateActiveException(); - if( !matcher.match( actualMessage ) ) { - data.resultType = ResultWas::ExpressionFailed; - data.reconstructedExpression = actualMessage; - } - AssertionResult result( m_assertionInfo, data ); - handleResult( result ); - } - - void ResultBuilder::captureExpression() { - AssertionResult result = build(); - handleResult( result ); - } - - void ResultBuilder::handleResult( AssertionResult const& result ) - { - getResultCapture().assertionEnded( result ); - - if( !result.isOk() ) { - if( getCurrentContext().getConfig()->shouldDebugBreak() ) - m_shouldDebugBreak = true; - if( getCurrentContext().getRunner()->aborting() || (m_assertionInfo.resultDisposition & ResultDisposition::Normal) ) - m_shouldThrow = true; - } - } - - void ResultBuilder::react() { -#if defined(CATCH_CONFIG_FAST_COMPILE) - if (m_shouldDebugBreak) { - /////////////////////////////////////////////////////////////////// - // To inspect the state during test, you need to go one level up the callstack - // To go back to the test and change execution, jump over the throw statement - /////////////////////////////////////////////////////////////////// - CATCH_BREAK_INTO_DEBUGGER(); - } -#endif - if( m_shouldThrow ) - throw Catch::TestFailureException(); - } - - bool ResultBuilder::shouldDebugBreak() const { return m_shouldDebugBreak; } - bool ResultBuilder::allowThrows() const { return getCurrentContext().getConfig()->allowThrows(); } - - AssertionResult ResultBuilder::build() const - { - return build( *this ); - } - - // CAVEAT: The returned AssertionResult stores a pointer to the argument expr, - // a temporary DecomposedExpression, which in turn holds references to - // operands, possibly temporary as well. - // It should immediately be passed to handleResult; if the expression - // needs to be reported, its string expansion must be composed before - // the temporaries are destroyed. - AssertionResult ResultBuilder::build( DecomposedExpression const& expr ) const - { - assert( m_data.resultType != ResultWas::Unknown ); - AssertionResultData data = m_data; - - if(m_usedStream) - data.message = m_stream().oss.str(); - data.decomposedExpression = &expr; // for lazy reconstruction - return AssertionResult( m_assertionInfo, data ); - } - - void ResultBuilder::reconstructExpression( std::string& dest ) const { - dest = capturedExpressionWithSecondArgument(m_assertionInfo.capturedExpression, m_assertionInfo.secondArg); - } - - void ResultBuilder::setExceptionGuard() { - m_guardException = true; - } - void ResultBuilder::unsetExceptionGuard() { - m_guardException = false; - } - -} // end namespace Catch - -// #included from: catch_tag_alias_registry.hpp -#define TWOBLUECUBES_CATCH_TAG_ALIAS_REGISTRY_HPP_INCLUDED - -namespace Catch { - - TagAliasRegistry::~TagAliasRegistry() {} - - Option TagAliasRegistry::find( std::string const& alias ) const { - std::map::const_iterator it = m_registry.find( alias ); - if( it != m_registry.end() ) - return it->second; - else - return Option(); - } - - std::string TagAliasRegistry::expandAliases( std::string const& unexpandedTestSpec ) const { - std::string expandedTestSpec = unexpandedTestSpec; - for( std::map::const_iterator it = m_registry.begin(), itEnd = m_registry.end(); - it != itEnd; - ++it ) { - std::size_t pos = expandedTestSpec.find( it->first ); - if( pos != std::string::npos ) { - expandedTestSpec = expandedTestSpec.substr( 0, pos ) + - it->second.tag + - expandedTestSpec.substr( pos + it->first.size() ); - } - } - return expandedTestSpec; - } - - void TagAliasRegistry::add( std::string const& alias, std::string const& tag, SourceLineInfo const& lineInfo ) { - - if( !startsWith( alias, "[@" ) || !endsWith( alias, ']' ) ) { - std::ostringstream oss; - oss << Colour( Colour::Red ) - << "error: tag alias, \"" << alias << "\" is not of the form [@alias name].\n" - << Colour( Colour::FileName ) - << lineInfo << '\n'; - throw std::domain_error( oss.str().c_str() ); - } - if( !m_registry.insert( std::make_pair( alias, TagAlias( tag, lineInfo ) ) ).second ) { - std::ostringstream oss; - oss << Colour( Colour::Red ) - << "error: tag alias, \"" << alias << "\" already registered.\n" - << "\tFirst seen at " - << Colour( Colour::Red ) << find(alias)->lineInfo << '\n' - << Colour( Colour::Red ) << "\tRedefined at " - << Colour( Colour::FileName) << lineInfo << '\n'; - throw std::domain_error( oss.str().c_str() ); - } - } - - ITagAliasRegistry::~ITagAliasRegistry() {} - - ITagAliasRegistry const& ITagAliasRegistry::get() { - return getRegistryHub().getTagAliasRegistry(); - } - - RegistrarForTagAliases::RegistrarForTagAliases( char const* alias, char const* tag, SourceLineInfo const& lineInfo ) { - getMutableRegistryHub().registerTagAlias( alias, tag, lineInfo ); - } - -} // end namespace Catch - -// #included from: catch_matchers_string.hpp - -namespace Catch { -namespace Matchers { - - namespace StdString { - - CasedString::CasedString( std::string const& str, CaseSensitive::Choice caseSensitivity ) - : m_caseSensitivity( caseSensitivity ), - m_str( adjustString( str ) ) - {} - std::string CasedString::adjustString( std::string const& str ) const { - return m_caseSensitivity == CaseSensitive::No - ? toLower( str ) - : str; - } - std::string CasedString::caseSensitivitySuffix() const { - return m_caseSensitivity == CaseSensitive::No - ? " (case insensitive)" - : std::string(); - } - - StringMatcherBase::StringMatcherBase( std::string const& operation, CasedString const& comparator ) - : m_comparator( comparator ), - m_operation( operation ) { - } - - std::string StringMatcherBase::describe() const { - std::string description; - description.reserve(5 + m_operation.size() + m_comparator.m_str.size() + - m_comparator.caseSensitivitySuffix().size()); - description += m_operation; - description += ": \""; - description += m_comparator.m_str; - description += "\""; - description += m_comparator.caseSensitivitySuffix(); - return description; - } - - EqualsMatcher::EqualsMatcher( CasedString const& comparator ) : StringMatcherBase( "equals", comparator ) {} - - bool EqualsMatcher::match( std::string const& source ) const { - return m_comparator.adjustString( source ) == m_comparator.m_str; - } - - ContainsMatcher::ContainsMatcher( CasedString const& comparator ) : StringMatcherBase( "contains", comparator ) {} - - bool ContainsMatcher::match( std::string const& source ) const { - return contains( m_comparator.adjustString( source ), m_comparator.m_str ); - } - - StartsWithMatcher::StartsWithMatcher( CasedString const& comparator ) : StringMatcherBase( "starts with", comparator ) {} - - bool StartsWithMatcher::match( std::string const& source ) const { - return startsWith( m_comparator.adjustString( source ), m_comparator.m_str ); - } - - EndsWithMatcher::EndsWithMatcher( CasedString const& comparator ) : StringMatcherBase( "ends with", comparator ) {} - - bool EndsWithMatcher::match( std::string const& source ) const { - return endsWith( m_comparator.adjustString( source ), m_comparator.m_str ); - } - - } // namespace StdString - - StdString::EqualsMatcher Equals( std::string const& str, CaseSensitive::Choice caseSensitivity ) { - return StdString::EqualsMatcher( StdString::CasedString( str, caseSensitivity) ); - } - StdString::ContainsMatcher Contains( std::string const& str, CaseSensitive::Choice caseSensitivity ) { - return StdString::ContainsMatcher( StdString::CasedString( str, caseSensitivity) ); - } - StdString::EndsWithMatcher EndsWith( std::string const& str, CaseSensitive::Choice caseSensitivity ) { - return StdString::EndsWithMatcher( StdString::CasedString( str, caseSensitivity) ); - } - StdString::StartsWithMatcher StartsWith( std::string const& str, CaseSensitive::Choice caseSensitivity ) { - return StdString::StartsWithMatcher( StdString::CasedString( str, caseSensitivity) ); - } - -} // namespace Matchers -} // namespace Catch -// #included from: ../reporters/catch_reporter_multi.hpp -#define TWOBLUECUBES_CATCH_REPORTER_MULTI_HPP_INCLUDED - -namespace Catch { - -class MultipleReporters : public SharedImpl { - typedef std::vector > Reporters; - Reporters m_reporters; - -public: - void add( Ptr const& reporter ) { - m_reporters.push_back( reporter ); - } - -public: // IStreamingReporter - - virtual ReporterPreferences getPreferences() const CATCH_OVERRIDE { - return m_reporters[0]->getPreferences(); - } - - virtual void noMatchingTestCases( std::string const& spec ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->noMatchingTestCases( spec ); - } - - virtual void testRunStarting( TestRunInfo const& testRunInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testRunStarting( testRunInfo ); - } - - virtual void testGroupStarting( GroupInfo const& groupInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testGroupStarting( groupInfo ); - } - - virtual void testCaseStarting( TestCaseInfo const& testInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testCaseStarting( testInfo ); - } - - virtual void sectionStarting( SectionInfo const& sectionInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->sectionStarting( sectionInfo ); - } - - virtual void assertionStarting( AssertionInfo const& assertionInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->assertionStarting( assertionInfo ); - } - - // The return value indicates if the messages buffer should be cleared: - virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - bool clearBuffer = false; - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - clearBuffer |= (*it)->assertionEnded( assertionStats ); - return clearBuffer; - } - - virtual void sectionEnded( SectionStats const& sectionStats ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->sectionEnded( sectionStats ); - } - - virtual void testCaseEnded( TestCaseStats const& testCaseStats ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testCaseEnded( testCaseStats ); - } - - virtual void testGroupEnded( TestGroupStats const& testGroupStats ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testGroupEnded( testGroupStats ); - } - - virtual void testRunEnded( TestRunStats const& testRunStats ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->testRunEnded( testRunStats ); - } - - virtual void skipTest( TestCaseInfo const& testInfo ) CATCH_OVERRIDE { - for( Reporters::const_iterator it = m_reporters.begin(), itEnd = m_reporters.end(); - it != itEnd; - ++it ) - (*it)->skipTest( testInfo ); - } - - virtual MultipleReporters* tryAsMulti() CATCH_OVERRIDE { - return this; - } - -}; - -Ptr addReporter( Ptr const& existingReporter, Ptr const& additionalReporter ) { - Ptr resultingReporter; - - if( existingReporter ) { - MultipleReporters* multi = existingReporter->tryAsMulti(); - if( !multi ) { - multi = new MultipleReporters; - resultingReporter = Ptr( multi ); - if( existingReporter ) - multi->add( existingReporter ); - } - else - resultingReporter = existingReporter; - multi->add( additionalReporter ); - } - else - resultingReporter = additionalReporter; - - return resultingReporter; -} - -} // end namespace Catch - -// #included from: ../reporters/catch_reporter_xml.hpp -#define TWOBLUECUBES_CATCH_REPORTER_XML_HPP_INCLUDED - -// #included from: catch_reporter_bases.hpp -#define TWOBLUECUBES_CATCH_REPORTER_BASES_HPP_INCLUDED - -#include -#include -#include -#include - -namespace Catch { - - namespace { - // Because formatting using c++ streams is stateful, drop down to C is required - // Alternatively we could use stringstream, but its performance is... not good. - std::string getFormattedDuration( double duration ) { - // Max exponent + 1 is required to represent the whole part - // + 1 for decimal point - // + 3 for the 3 decimal places - // + 1 for null terminator - const size_t maxDoubleSize = DBL_MAX_10_EXP + 1 + 1 + 3 + 1; - char buffer[maxDoubleSize]; - - // Save previous errno, to prevent sprintf from overwriting it - ErrnoGuard guard; -#ifdef _MSC_VER - sprintf_s(buffer, "%.3f", duration); -#else - sprintf(buffer, "%.3f", duration); -#endif - return std::string(buffer); - } - } - - struct StreamingReporterBase : SharedImpl { - - StreamingReporterBase( ReporterConfig const& _config ) - : m_config( _config.fullConfig() ), - stream( _config.stream() ) - { - m_reporterPrefs.shouldRedirectStdOut = false; - } - - virtual ReporterPreferences getPreferences() const CATCH_OVERRIDE { - return m_reporterPrefs; - } - - virtual ~StreamingReporterBase() CATCH_OVERRIDE; - - virtual void noMatchingTestCases( std::string const& ) CATCH_OVERRIDE {} - - virtual void testRunStarting( TestRunInfo const& _testRunInfo ) CATCH_OVERRIDE { - currentTestRunInfo = _testRunInfo; - } - virtual void testGroupStarting( GroupInfo const& _groupInfo ) CATCH_OVERRIDE { - currentGroupInfo = _groupInfo; - } - - virtual void testCaseStarting( TestCaseInfo const& _testInfo ) CATCH_OVERRIDE { - currentTestCaseInfo = _testInfo; - } - virtual void sectionStarting( SectionInfo const& _sectionInfo ) CATCH_OVERRIDE { - m_sectionStack.push_back( _sectionInfo ); - } - - virtual void sectionEnded( SectionStats const& /* _sectionStats */ ) CATCH_OVERRIDE { - m_sectionStack.pop_back(); - } - virtual void testCaseEnded( TestCaseStats const& /* _testCaseStats */ ) CATCH_OVERRIDE { - currentTestCaseInfo.reset(); - } - virtual void testGroupEnded( TestGroupStats const& /* _testGroupStats */ ) CATCH_OVERRIDE { - currentGroupInfo.reset(); - } - virtual void testRunEnded( TestRunStats const& /* _testRunStats */ ) CATCH_OVERRIDE { - currentTestCaseInfo.reset(); - currentGroupInfo.reset(); - currentTestRunInfo.reset(); - } - - virtual void skipTest( TestCaseInfo const& ) CATCH_OVERRIDE { - // Don't do anything with this by default. - // It can optionally be overridden in the derived class. - } - - Ptr m_config; - std::ostream& stream; - - LazyStat currentTestRunInfo; - LazyStat currentGroupInfo; - LazyStat currentTestCaseInfo; - - std::vector m_sectionStack; - ReporterPreferences m_reporterPrefs; - }; - - struct CumulativeReporterBase : SharedImpl { - template - struct Node : SharedImpl<> { - explicit Node( T const& _value ) : value( _value ) {} - virtual ~Node() {} - - typedef std::vector > ChildNodes; - T value; - ChildNodes children; - }; - struct SectionNode : SharedImpl<> { - explicit SectionNode( SectionStats const& _stats ) : stats( _stats ) {} - virtual ~SectionNode(); - - bool operator == ( SectionNode const& other ) const { - return stats.sectionInfo.lineInfo == other.stats.sectionInfo.lineInfo; - } - bool operator == ( Ptr const& other ) const { - return operator==( *other ); - } - - SectionStats stats; - typedef std::vector > ChildSections; - typedef std::vector Assertions; - ChildSections childSections; - Assertions assertions; - std::string stdOut; - std::string stdErr; - }; - - struct BySectionInfo { - BySectionInfo( SectionInfo const& other ) : m_other( other ) {} - BySectionInfo( BySectionInfo const& other ) : m_other( other.m_other ) {} - bool operator() ( Ptr const& node ) const { - return ((node->stats.sectionInfo.name == m_other.name) && - (node->stats.sectionInfo.lineInfo == m_other.lineInfo)); - } - private: - void operator=( BySectionInfo const& ); - SectionInfo const& m_other; - }; - - typedef Node TestCaseNode; - typedef Node TestGroupNode; - typedef Node TestRunNode; - - CumulativeReporterBase( ReporterConfig const& _config ) - : m_config( _config.fullConfig() ), - stream( _config.stream() ) - { - m_reporterPrefs.shouldRedirectStdOut = false; - } - ~CumulativeReporterBase(); - - virtual ReporterPreferences getPreferences() const CATCH_OVERRIDE { - return m_reporterPrefs; - } - - virtual void testRunStarting( TestRunInfo const& ) CATCH_OVERRIDE {} - virtual void testGroupStarting( GroupInfo const& ) CATCH_OVERRIDE {} - - virtual void testCaseStarting( TestCaseInfo const& ) CATCH_OVERRIDE {} - - virtual void sectionStarting( SectionInfo const& sectionInfo ) CATCH_OVERRIDE { - SectionStats incompleteStats( sectionInfo, Counts(), 0, false ); - Ptr node; - if( m_sectionStack.empty() ) { - if( !m_rootSection ) - m_rootSection = new SectionNode( incompleteStats ); - node = m_rootSection; - } - else { - SectionNode& parentNode = *m_sectionStack.back(); - SectionNode::ChildSections::const_iterator it = - std::find_if( parentNode.childSections.begin(), - parentNode.childSections.end(), - BySectionInfo( sectionInfo ) ); - if( it == parentNode.childSections.end() ) { - node = new SectionNode( incompleteStats ); - parentNode.childSections.push_back( node ); - } - else - node = *it; - } - m_sectionStack.push_back( node ); - m_deepestSection = node; - } - - virtual void assertionStarting( AssertionInfo const& ) CATCH_OVERRIDE {} - - virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - assert( !m_sectionStack.empty() ); - SectionNode& sectionNode = *m_sectionStack.back(); - sectionNode.assertions.push_back( assertionStats ); - // AssertionResult holds a pointer to a temporary DecomposedExpression, - // which getExpandedExpression() calls to build the expression string. - // Our section stack copy of the assertionResult will likely outlive the - // temporary, so it must be expanded or discarded now to avoid calling - // a destroyed object later. - prepareExpandedExpression( sectionNode.assertions.back().assertionResult ); - return true; - } - virtual void sectionEnded( SectionStats const& sectionStats ) CATCH_OVERRIDE { - assert( !m_sectionStack.empty() ); - SectionNode& node = *m_sectionStack.back(); - node.stats = sectionStats; - m_sectionStack.pop_back(); - } - virtual void testCaseEnded( TestCaseStats const& testCaseStats ) CATCH_OVERRIDE { - Ptr node = new TestCaseNode( testCaseStats ); - assert( m_sectionStack.size() == 0 ); - node->children.push_back( m_rootSection ); - m_testCases.push_back( node ); - m_rootSection.reset(); - - assert( m_deepestSection ); - m_deepestSection->stdOut = testCaseStats.stdOut; - m_deepestSection->stdErr = testCaseStats.stdErr; - } - virtual void testGroupEnded( TestGroupStats const& testGroupStats ) CATCH_OVERRIDE { - Ptr node = new TestGroupNode( testGroupStats ); - node->children.swap( m_testCases ); - m_testGroups.push_back( node ); - } - virtual void testRunEnded( TestRunStats const& testRunStats ) CATCH_OVERRIDE { - Ptr node = new TestRunNode( testRunStats ); - node->children.swap( m_testGroups ); - m_testRuns.push_back( node ); - testRunEndedCumulative(); - } - virtual void testRunEndedCumulative() = 0; - - virtual void skipTest( TestCaseInfo const& ) CATCH_OVERRIDE {} - - virtual void prepareExpandedExpression( AssertionResult& result ) const { - if( result.isOk() ) - result.discardDecomposedExpression(); - else - result.expandDecomposedExpression(); - } - - Ptr m_config; - std::ostream& stream; - std::vector m_assertions; - std::vector > > m_sections; - std::vector > m_testCases; - std::vector > m_testGroups; - - std::vector > m_testRuns; - - Ptr m_rootSection; - Ptr m_deepestSection; - std::vector > m_sectionStack; - ReporterPreferences m_reporterPrefs; - - }; - - template - char const* getLineOfChars() { - static char line[CATCH_CONFIG_CONSOLE_WIDTH] = {0}; - if( !*line ) { - std::memset( line, C, CATCH_CONFIG_CONSOLE_WIDTH-1 ); - line[CATCH_CONFIG_CONSOLE_WIDTH-1] = 0; - } - return line; - } - - struct TestEventListenerBase : StreamingReporterBase { - TestEventListenerBase( ReporterConfig const& _config ) - : StreamingReporterBase( _config ) - {} - - virtual void assertionStarting( AssertionInfo const& ) CATCH_OVERRIDE {} - virtual bool assertionEnded( AssertionStats const& ) CATCH_OVERRIDE { - return false; - } - }; - -} // end namespace Catch - -// #included from: ../internal/catch_reporter_registrars.hpp -#define TWOBLUECUBES_CATCH_REPORTER_REGISTRARS_HPP_INCLUDED - -namespace Catch { - - template - class LegacyReporterRegistrar { - - class ReporterFactory : public IReporterFactory { - virtual IStreamingReporter* create( ReporterConfig const& config ) const { - return new LegacyReporterAdapter( new T( config ) ); - } - - virtual std::string getDescription() const { - return T::getDescription(); - } - }; - - public: - - LegacyReporterRegistrar( std::string const& name ) { - getMutableRegistryHub().registerReporter( name, new ReporterFactory() ); - } - }; - - template - class ReporterRegistrar { - - class ReporterFactory : public SharedImpl { - - // *** Please Note ***: - // - If you end up here looking at a compiler error because it's trying to register - // your custom reporter class be aware that the native reporter interface has changed - // to IStreamingReporter. The "legacy" interface, IReporter, is still supported via - // an adapter. Just use REGISTER_LEGACY_REPORTER to take advantage of the adapter. - // However please consider updating to the new interface as the old one is now - // deprecated and will probably be removed quite soon! - // Please contact me via github if you have any questions at all about this. - // In fact, ideally, please contact me anyway to let me know you've hit this - as I have - // no idea who is actually using custom reporters at all (possibly no-one!). - // The new interface is designed to minimise exposure to interface changes in the future. - virtual IStreamingReporter* create( ReporterConfig const& config ) const { - return new T( config ); - } - - virtual std::string getDescription() const { - return T::getDescription(); - } - }; - - public: - - ReporterRegistrar( std::string const& name ) { - getMutableRegistryHub().registerReporter( name, new ReporterFactory() ); - } - }; - - template - class ListenerRegistrar { - - class ListenerFactory : public SharedImpl { - - virtual IStreamingReporter* create( ReporterConfig const& config ) const { - return new T( config ); - } - virtual std::string getDescription() const { - return std::string(); - } - }; - - public: - - ListenerRegistrar() { - getMutableRegistryHub().registerListener( new ListenerFactory() ); - } - }; -} - -#define INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) \ - namespace{ Catch::LegacyReporterRegistrar catch_internal_RegistrarFor##reporterType( name ); } - -#define INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType ) \ - namespace{ Catch::ReporterRegistrar catch_internal_RegistrarFor##reporterType( name ); } - -// Deprecated - use the form without INTERNAL_ -#define INTERNAL_CATCH_REGISTER_LISTENER( listenerType ) \ - namespace{ Catch::ListenerRegistrar catch_internal_RegistrarFor##listenerType; } - -#define CATCH_REGISTER_LISTENER( listenerType ) \ - namespace{ Catch::ListenerRegistrar catch_internal_RegistrarFor##listenerType; } - -// #included from: ../internal/catch_xmlwriter.hpp -#define TWOBLUECUBES_CATCH_XMLWRITER_HPP_INCLUDED - -#include -#include -#include -#include - -namespace Catch { - - class XmlEncode { - public: - enum ForWhat { ForTextNodes, ForAttributes }; - - XmlEncode( std::string const& str, ForWhat forWhat = ForTextNodes ) - : m_str( str ), - m_forWhat( forWhat ) - {} - - void encodeTo( std::ostream& os ) const { - - // Apostrophe escaping not necessary if we always use " to write attributes - // (see: http://www.w3.org/TR/xml/#syntax) - - for( std::size_t i = 0; i < m_str.size(); ++ i ) { - char c = m_str[i]; - switch( c ) { - case '<': os << "<"; break; - case '&': os << "&"; break; - - case '>': - // See: http://www.w3.org/TR/xml/#syntax - if( i > 2 && m_str[i-1] == ']' && m_str[i-2] == ']' ) - os << ">"; - else - os << c; - break; - - case '\"': - if( m_forWhat == ForAttributes ) - os << """; - else - os << c; - break; - - default: - // Escape control chars - based on contribution by @espenalb in PR #465 and - // by @mrpi PR #588 - if ( ( c >= 0 && c < '\x09' ) || ( c > '\x0D' && c < '\x20') || c=='\x7F' ) { - // see http://stackoverflow.com/questions/404107/why-are-control-characters-illegal-in-xml-1-0 - os << "\\x" << std::uppercase << std::hex << std::setfill('0') << std::setw(2) - << static_cast( c ); - } - else - os << c; - } - } - } - - friend std::ostream& operator << ( std::ostream& os, XmlEncode const& xmlEncode ) { - xmlEncode.encodeTo( os ); - return os; - } - - private: - std::string m_str; - ForWhat m_forWhat; - }; - - class XmlWriter { - public: - - class ScopedElement { - public: - ScopedElement( XmlWriter* writer ) - : m_writer( writer ) - {} - - ScopedElement( ScopedElement const& other ) - : m_writer( other.m_writer ){ - other.m_writer = CATCH_NULL; - } - - ~ScopedElement() { - if( m_writer ) - m_writer->endElement(); - } - - ScopedElement& writeText( std::string const& text, bool indent = true ) { - m_writer->writeText( text, indent ); - return *this; - } - - template - ScopedElement& writeAttribute( std::string const& name, T const& attribute ) { - m_writer->writeAttribute( name, attribute ); - return *this; - } - - private: - mutable XmlWriter* m_writer; - }; - - XmlWriter() - : m_tagIsOpen( false ), - m_needsNewline( false ), - m_os( Catch::cout() ) - { - writeDeclaration(); - } - - XmlWriter( std::ostream& os ) - : m_tagIsOpen( false ), - m_needsNewline( false ), - m_os( os ) - { - writeDeclaration(); - } - - ~XmlWriter() { - while( !m_tags.empty() ) - endElement(); - } - - XmlWriter& startElement( std::string const& name ) { - ensureTagClosed(); - newlineIfNecessary(); - m_os << m_indent << '<' << name; - m_tags.push_back( name ); - m_indent += " "; - m_tagIsOpen = true; - return *this; - } - - ScopedElement scopedElement( std::string const& name ) { - ScopedElement scoped( this ); - startElement( name ); - return scoped; - } - - XmlWriter& endElement() { - newlineIfNecessary(); - m_indent = m_indent.substr( 0, m_indent.size()-2 ); - if( m_tagIsOpen ) { - m_os << "/>"; - m_tagIsOpen = false; - } - else { - m_os << m_indent << ""; - } - m_os << std::endl; - m_tags.pop_back(); - return *this; - } - - XmlWriter& writeAttribute( std::string const& name, std::string const& attribute ) { - if( !name.empty() && !attribute.empty() ) - m_os << ' ' << name << "=\"" << XmlEncode( attribute, XmlEncode::ForAttributes ) << '"'; - return *this; - } - - XmlWriter& writeAttribute( std::string const& name, bool attribute ) { - m_os << ' ' << name << "=\"" << ( attribute ? "true" : "false" ) << '"'; - return *this; - } - - template - XmlWriter& writeAttribute( std::string const& name, T const& attribute ) { - std::ostringstream oss; - oss << attribute; - return writeAttribute( name, oss.str() ); - } - - XmlWriter& writeText( std::string const& text, bool indent = true ) { - if( !text.empty() ){ - bool tagWasOpen = m_tagIsOpen; - ensureTagClosed(); - if( tagWasOpen && indent ) - m_os << m_indent; - m_os << XmlEncode( text ); - m_needsNewline = true; - } - return *this; - } - - XmlWriter& writeComment( std::string const& text ) { - ensureTagClosed(); - m_os << m_indent << ""; - m_needsNewline = true; - return *this; - } - - void writeStylesheetRef( std::string const& url ) { - m_os << "\n"; - } - - XmlWriter& writeBlankLine() { - ensureTagClosed(); - m_os << '\n'; - return *this; - } - - void ensureTagClosed() { - if( m_tagIsOpen ) { - m_os << ">" << std::endl; - m_tagIsOpen = false; - } - } - - private: - XmlWriter( XmlWriter const& ); - void operator=( XmlWriter const& ); - - void writeDeclaration() { - m_os << "\n"; - } - - void newlineIfNecessary() { - if( m_needsNewline ) { - m_os << std::endl; - m_needsNewline = false; - } - } - - bool m_tagIsOpen; - bool m_needsNewline; - std::vector m_tags; - std::string m_indent; - std::ostream& m_os; - }; - -} - -namespace Catch { - class XmlReporter : public StreamingReporterBase { - public: - XmlReporter( ReporterConfig const& _config ) - : StreamingReporterBase( _config ), - m_xml(_config.stream()), - m_sectionDepth( 0 ) - { - m_reporterPrefs.shouldRedirectStdOut = true; - } - - virtual ~XmlReporter() CATCH_OVERRIDE; - - static std::string getDescription() { - return "Reports test results as an XML document"; - } - - virtual std::string getStylesheetRef() const { - return std::string(); - } - - void writeSourceInfo( SourceLineInfo const& sourceInfo ) { - m_xml - .writeAttribute( "filename", sourceInfo.file ) - .writeAttribute( "line", sourceInfo.line ); - } - - public: // StreamingReporterBase - - virtual void noMatchingTestCases( std::string const& s ) CATCH_OVERRIDE { - StreamingReporterBase::noMatchingTestCases( s ); - } - - virtual void testRunStarting( TestRunInfo const& testInfo ) CATCH_OVERRIDE { - StreamingReporterBase::testRunStarting( testInfo ); - std::string stylesheetRef = getStylesheetRef(); - if( !stylesheetRef.empty() ) - m_xml.writeStylesheetRef( stylesheetRef ); - m_xml.startElement( "Catch" ); - if( !m_config->name().empty() ) - m_xml.writeAttribute( "name", m_config->name() ); - } - - virtual void testGroupStarting( GroupInfo const& groupInfo ) CATCH_OVERRIDE { - StreamingReporterBase::testGroupStarting( groupInfo ); - m_xml.startElement( "Group" ) - .writeAttribute( "name", groupInfo.name ); - } - - virtual void testCaseStarting( TestCaseInfo const& testInfo ) CATCH_OVERRIDE { - StreamingReporterBase::testCaseStarting(testInfo); - m_xml.startElement( "TestCase" ) - .writeAttribute( "name", trim( testInfo.name ) ) - .writeAttribute( "description", testInfo.description ) - .writeAttribute( "tags", testInfo.tagsAsString ); - - writeSourceInfo( testInfo.lineInfo ); - - if ( m_config->showDurations() == ShowDurations::Always ) - m_testCaseTimer.start(); - m_xml.ensureTagClosed(); - } - - virtual void sectionStarting( SectionInfo const& sectionInfo ) CATCH_OVERRIDE { - StreamingReporterBase::sectionStarting( sectionInfo ); - if( m_sectionDepth++ > 0 ) { - m_xml.startElement( "Section" ) - .writeAttribute( "name", trim( sectionInfo.name ) ) - .writeAttribute( "description", sectionInfo.description ); - writeSourceInfo( sectionInfo.lineInfo ); - m_xml.ensureTagClosed(); - } - } - - virtual void assertionStarting( AssertionInfo const& ) CATCH_OVERRIDE { } - - virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - - AssertionResult const& result = assertionStats.assertionResult; - - bool includeResults = m_config->includeSuccessfulResults() || !result.isOk(); - - if( includeResults || result.getResultType() == ResultWas::Warning ) { - // Print any info messages in tags. - for( std::vector::const_iterator it = assertionStats.infoMessages.begin(), itEnd = assertionStats.infoMessages.end(); - it != itEnd; - ++it ) { - if( it->type == ResultWas::Info && includeResults ) { - m_xml.scopedElement( "Info" ) - .writeText( it->message ); - } else if ( it->type == ResultWas::Warning ) { - m_xml.scopedElement( "Warning" ) - .writeText( it->message ); - } - } - } - - // Drop out if result was successful but we're not printing them. - if( !includeResults && result.getResultType() != ResultWas::Warning ) - return true; - - // Print the expression if there is one. - if( result.hasExpression() ) { - m_xml.startElement( "Expression" ) - .writeAttribute( "success", result.succeeded() ) - .writeAttribute( "type", result.getTestMacroName() ); - - writeSourceInfo( result.getSourceInfo() ); - - m_xml.scopedElement( "Original" ) - .writeText( result.getExpression() ); - m_xml.scopedElement( "Expanded" ) - .writeText( result.getExpandedExpression() ); - } - - // And... Print a result applicable to each result type. - switch( result.getResultType() ) { - case ResultWas::ThrewException: - m_xml.startElement( "Exception" ); - writeSourceInfo( result.getSourceInfo() ); - m_xml.writeText( result.getMessage() ); - m_xml.endElement(); - break; - case ResultWas::FatalErrorCondition: - m_xml.startElement( "FatalErrorCondition" ); - writeSourceInfo( result.getSourceInfo() ); - m_xml.writeText( result.getMessage() ); - m_xml.endElement(); - break; - case ResultWas::Info: - m_xml.scopedElement( "Info" ) - .writeText( result.getMessage() ); - break; - case ResultWas::Warning: - // Warning will already have been written - break; - case ResultWas::ExplicitFailure: - m_xml.startElement( "Failure" ); - writeSourceInfo( result.getSourceInfo() ); - m_xml.writeText( result.getMessage() ); - m_xml.endElement(); - break; - default: - break; - } - - if( result.hasExpression() ) - m_xml.endElement(); - - return true; - } - - virtual void sectionEnded( SectionStats const& sectionStats ) CATCH_OVERRIDE { - StreamingReporterBase::sectionEnded( sectionStats ); - if( --m_sectionDepth > 0 ) { - XmlWriter::ScopedElement e = m_xml.scopedElement( "OverallResults" ); - e.writeAttribute( "successes", sectionStats.assertions.passed ); - e.writeAttribute( "failures", sectionStats.assertions.failed ); - e.writeAttribute( "expectedFailures", sectionStats.assertions.failedButOk ); - - if ( m_config->showDurations() == ShowDurations::Always ) - e.writeAttribute( "durationInSeconds", sectionStats.durationInSeconds ); - - m_xml.endElement(); - } - } - - virtual void testCaseEnded( TestCaseStats const& testCaseStats ) CATCH_OVERRIDE { - StreamingReporterBase::testCaseEnded( testCaseStats ); - XmlWriter::ScopedElement e = m_xml.scopedElement( "OverallResult" ); - e.writeAttribute( "success", testCaseStats.totals.assertions.allOk() ); - - if ( m_config->showDurations() == ShowDurations::Always ) - e.writeAttribute( "durationInSeconds", m_testCaseTimer.getElapsedSeconds() ); - - if( !testCaseStats.stdOut.empty() ) - m_xml.scopedElement( "StdOut" ).writeText( trim( testCaseStats.stdOut ), false ); - if( !testCaseStats.stdErr.empty() ) - m_xml.scopedElement( "StdErr" ).writeText( trim( testCaseStats.stdErr ), false ); - - m_xml.endElement(); - } - - virtual void testGroupEnded( TestGroupStats const& testGroupStats ) CATCH_OVERRIDE { - StreamingReporterBase::testGroupEnded( testGroupStats ); - // TODO: Check testGroupStats.aborting and act accordingly. - m_xml.scopedElement( "OverallResults" ) - .writeAttribute( "successes", testGroupStats.totals.assertions.passed ) - .writeAttribute( "failures", testGroupStats.totals.assertions.failed ) - .writeAttribute( "expectedFailures", testGroupStats.totals.assertions.failedButOk ); - m_xml.endElement(); - } - - virtual void testRunEnded( TestRunStats const& testRunStats ) CATCH_OVERRIDE { - StreamingReporterBase::testRunEnded( testRunStats ); - m_xml.scopedElement( "OverallResults" ) - .writeAttribute( "successes", testRunStats.totals.assertions.passed ) - .writeAttribute( "failures", testRunStats.totals.assertions.failed ) - .writeAttribute( "expectedFailures", testRunStats.totals.assertions.failedButOk ); - m_xml.endElement(); - } - - private: - Timer m_testCaseTimer; - XmlWriter m_xml; - int m_sectionDepth; - }; - - INTERNAL_CATCH_REGISTER_REPORTER( "xml", XmlReporter ) - -} // end namespace Catch - -// #included from: ../reporters/catch_reporter_junit.hpp -#define TWOBLUECUBES_CATCH_REPORTER_JUNIT_HPP_INCLUDED - -#include - -namespace Catch { - - namespace { - std::string getCurrentTimestamp() { - // Beware, this is not reentrant because of backward compatibility issues - // Also, UTC only, again because of backward compatibility (%z is C++11) - time_t rawtime; - std::time(&rawtime); - const size_t timeStampSize = sizeof("2017-01-16T17:06:45Z"); - -#ifdef _MSC_VER - std::tm timeInfo = {}; - gmtime_s(&timeInfo, &rawtime); -#else - std::tm* timeInfo; - timeInfo = std::gmtime(&rawtime); -#endif - - char timeStamp[timeStampSize]; - const char * const fmt = "%Y-%m-%dT%H:%M:%SZ"; - -#ifdef _MSC_VER - std::strftime(timeStamp, timeStampSize, fmt, &timeInfo); -#else - std::strftime(timeStamp, timeStampSize, fmt, timeInfo); -#endif - return std::string(timeStamp); - } - - } - - class JunitReporter : public CumulativeReporterBase { - public: - JunitReporter( ReporterConfig const& _config ) - : CumulativeReporterBase( _config ), - xml( _config.stream() ), - unexpectedExceptions( 0 ), - m_okToFail( false ) - { - m_reporterPrefs.shouldRedirectStdOut = true; - } - - virtual ~JunitReporter() CATCH_OVERRIDE; - - static std::string getDescription() { - return "Reports test results in an XML format that looks like Ant's junitreport target"; - } - - virtual void noMatchingTestCases( std::string const& /*spec*/ ) CATCH_OVERRIDE {} - - virtual void testRunStarting( TestRunInfo const& runInfo ) CATCH_OVERRIDE { - CumulativeReporterBase::testRunStarting( runInfo ); - xml.startElement( "testsuites" ); - } - - virtual void testGroupStarting( GroupInfo const& groupInfo ) CATCH_OVERRIDE { - suiteTimer.start(); - stdOutForSuite.str(""); - stdErrForSuite.str(""); - unexpectedExceptions = 0; - CumulativeReporterBase::testGroupStarting( groupInfo ); - } - - virtual void testCaseStarting( TestCaseInfo const& testCaseInfo ) CATCH_OVERRIDE { - m_okToFail = testCaseInfo.okToFail(); - } - virtual bool assertionEnded( AssertionStats const& assertionStats ) CATCH_OVERRIDE { - if( assertionStats.assertionResult.getResultType() == ResultWas::ThrewException && !m_okToFail ) - unexpectedExceptions++; - return CumulativeReporterBase::assertionEnded( assertionStats ); - } - - virtual void testCaseEnded( TestCaseStats const& testCaseStats ) CATCH_OVERRIDE { - stdOutForSuite << testCaseStats.stdOut; - stdErrForSuite << testCaseStats.stdErr; - CumulativeReporterBase::testCaseEnded( testCaseStats ); - } - - virtual void testGroupEnded( TestGroupStats const& testGroupStats ) CATCH_OVERRIDE { - double suiteTime = suiteTimer.getElapsedSeconds(); - CumulativeReporterBase::testGroupEnded( testGroupStats ); - writeGroup( *m_testGroups.back(), suiteTime ); - } - - virtual void testRunEndedCumulative() CATCH_OVERRIDE { - xml.endElement(); - } - - void writeGroup( TestGroupNode const& groupNode, double suiteTime ) { - XmlWriter::ScopedElement e = xml.scopedElement( "testsuite" ); - TestGroupStats const& stats = groupNode.value; - xml.writeAttribute( "name", stats.groupInfo.name ); - xml.writeAttribute( "errors", unexpectedExceptions ); - xml.writeAttribute( "failures", stats.totals.assertions.failed-unexpectedExceptions ); - xml.writeAttribute( "tests", stats.totals.assertions.total() ); - xml.writeAttribute( "hostname", "tbd" ); // !TBD - if( m_config->showDurations() == ShowDurations::Never ) - xml.writeAttribute( "time", "" ); - else - xml.writeAttribute( "time", suiteTime ); - xml.writeAttribute( "timestamp", getCurrentTimestamp() ); - - // Write test cases - for( TestGroupNode::ChildNodes::const_iterator - it = groupNode.children.begin(), itEnd = groupNode.children.end(); - it != itEnd; - ++it ) - writeTestCase( **it ); - - xml.scopedElement( "system-out" ).writeText( trim( stdOutForSuite.str() ), false ); - xml.scopedElement( "system-err" ).writeText( trim( stdErrForSuite.str() ), false ); - } - - void writeTestCase( TestCaseNode const& testCaseNode ) { - TestCaseStats const& stats = testCaseNode.value; - - // All test cases have exactly one section - which represents the - // test case itself. That section may have 0-n nested sections - assert( testCaseNode.children.size() == 1 ); - SectionNode const& rootSection = *testCaseNode.children.front(); - - std::string className = stats.testInfo.className; - - if( className.empty() ) { - if( rootSection.childSections.empty() ) - className = "global"; - } - writeSection( className, "", rootSection ); - } - - void writeSection( std::string const& className, - std::string const& rootName, - SectionNode const& sectionNode ) { - std::string name = trim( sectionNode.stats.sectionInfo.name ); - if( !rootName.empty() ) - name = rootName + '/' + name; - - if( !sectionNode.assertions.empty() || - !sectionNode.stdOut.empty() || - !sectionNode.stdErr.empty() ) { - XmlWriter::ScopedElement e = xml.scopedElement( "testcase" ); - if( className.empty() ) { - xml.writeAttribute( "classname", name ); - xml.writeAttribute( "name", "root" ); - } - else { - xml.writeAttribute( "classname", className ); - xml.writeAttribute( "name", name ); - } - xml.writeAttribute( "time", Catch::toString( sectionNode.stats.durationInSeconds ) ); - - writeAssertions( sectionNode ); - - if( !sectionNode.stdOut.empty() ) - xml.scopedElement( "system-out" ).writeText( trim( sectionNode.stdOut ), false ); - if( !sectionNode.stdErr.empty() ) - xml.scopedElement( "system-err" ).writeText( trim( sectionNode.stdErr ), false ); - } - for( SectionNode::ChildSections::const_iterator - it = sectionNode.childSections.begin(), - itEnd = sectionNode.childSections.end(); - it != itEnd; - ++it ) - if( className.empty() ) - writeSection( name, "", **it ); - else - writeSection( className, name, **it ); - } - - void writeAssertions( SectionNode const& sectionNode ) { - for( SectionNode::Assertions::const_iterator - it = sectionNode.assertions.begin(), itEnd = sectionNode.assertions.end(); - it != itEnd; - ++it ) - writeAssertion( *it ); - } - void writeAssertion( AssertionStats const& stats ) { - AssertionResult const& result = stats.assertionResult; - if( !result.isOk() ) { - std::string elementName; - switch( result.getResultType() ) { - case ResultWas::ThrewException: - case ResultWas::FatalErrorCondition: - elementName = "error"; - break; - case ResultWas::ExplicitFailure: - elementName = "failure"; - break; - case ResultWas::ExpressionFailed: - elementName = "failure"; - break; - case ResultWas::DidntThrowException: - elementName = "failure"; - break; - - // We should never see these here: - case ResultWas::Info: - case ResultWas::Warning: - case ResultWas::Ok: - case ResultWas::Unknown: - case ResultWas::FailureBit: - case ResultWas::Exception: - elementName = "internalError"; - break; - } - - XmlWriter::ScopedElement e = xml.scopedElement( elementName ); - - xml.writeAttribute( "message", result.getExpandedExpression() ); - xml.writeAttribute( "type", result.getTestMacroName() ); - - std::ostringstream oss; - if( !result.getMessage().empty() ) - oss << result.getMessage() << '\n'; - for( std::vector::const_iterator - it = stats.infoMessages.begin(), - itEnd = stats.infoMessages.end(); - it != itEnd; - ++it ) - if( it->type == ResultWas::Info ) - oss << it->message << '\n'; - - oss << "at " << result.getSourceInfo(); - xml.writeText( oss.str(), false ); - } - } - - XmlWriter xml; - Timer suiteTimer; - std::ostringstream stdOutForSuite; - std::ostringstream stdErrForSuite; - unsigned int unexpectedExceptions; - bool m_okToFail; - }; - - INTERNAL_CATCH_REGISTER_REPORTER( "junit", JunitReporter ) - -} // end namespace Catch - -// #included from: ../reporters/catch_reporter_console.hpp -#define TWOBLUECUBES_CATCH_REPORTER_CONSOLE_HPP_INCLUDED - -#include -#include -#include - -namespace Catch { - - struct ConsoleReporter : StreamingReporterBase { - ConsoleReporter( ReporterConfig const& _config ) - : StreamingReporterBase( _config ), - m_headerPrinted( false ) - {} - - virtual ~ConsoleReporter() CATCH_OVERRIDE; - static std::string getDescription() { - return "Reports test results as plain lines of text"; - } - - virtual void noMatchingTestCases( std::string const& spec ) CATCH_OVERRIDE { - stream << "No test cases matched '" << spec << '\'' << std::endl; - } - - virtual void assertionStarting( AssertionInfo const& ) CATCH_OVERRIDE { - } - - virtual bool assertionEnded( AssertionStats const& _assertionStats ) CATCH_OVERRIDE { - AssertionResult const& result = _assertionStats.assertionResult; - - bool includeResults = m_config->includeSuccessfulResults() || !result.isOk(); - - // Drop out if result was successful but we're not printing them. - if( !includeResults && result.getResultType() != ResultWas::Warning ) - return false; - - lazyPrint(); - - AssertionPrinter printer( stream, _assertionStats, includeResults ); - printer.print(); - stream << std::endl; - return true; - } - - virtual void sectionStarting( SectionInfo const& _sectionInfo ) CATCH_OVERRIDE { - m_headerPrinted = false; - StreamingReporterBase::sectionStarting( _sectionInfo ); - } - virtual void sectionEnded( SectionStats const& _sectionStats ) CATCH_OVERRIDE { - if( _sectionStats.missingAssertions ) { - lazyPrint(); - Colour colour( Colour::ResultError ); - if( m_sectionStack.size() > 1 ) - stream << "\nNo assertions in section"; - else - stream << "\nNo assertions in test case"; - stream << " '" << _sectionStats.sectionInfo.name << "'\n" << std::endl; - } - if( m_config->showDurations() == ShowDurations::Always ) { - stream << getFormattedDuration(_sectionStats.durationInSeconds) << " s: " << _sectionStats.sectionInfo.name << std::endl; - } - if( m_headerPrinted ) { - m_headerPrinted = false; - } - StreamingReporterBase::sectionEnded( _sectionStats ); - } - - virtual void testCaseEnded( TestCaseStats const& _testCaseStats ) CATCH_OVERRIDE { - StreamingReporterBase::testCaseEnded( _testCaseStats ); - m_headerPrinted = false; - } - virtual void testGroupEnded( TestGroupStats const& _testGroupStats ) CATCH_OVERRIDE { - if( currentGroupInfo.used ) { - printSummaryDivider(); - stream << "Summary for group '" << _testGroupStats.groupInfo.name << "':\n"; - printTotals( _testGroupStats.totals ); - stream << '\n' << std::endl; - } - StreamingReporterBase::testGroupEnded( _testGroupStats ); - } - virtual void testRunEnded( TestRunStats const& _testRunStats ) CATCH_OVERRIDE { - printTotalsDivider( _testRunStats.totals ); - printTotals( _testRunStats.totals ); - stream << std::endl; - StreamingReporterBase::testRunEnded( _testRunStats ); - } - - private: - - class AssertionPrinter { - void operator= ( AssertionPrinter const& ); - public: - AssertionPrinter( std::ostream& _stream, AssertionStats const& _stats, bool _printInfoMessages ) - : stream( _stream ), - stats( _stats ), - result( _stats.assertionResult ), - colour( Colour::None ), - message( result.getMessage() ), - messages( _stats.infoMessages ), - printInfoMessages( _printInfoMessages ) - { - switch( result.getResultType() ) { - case ResultWas::Ok: - colour = Colour::Success; - passOrFail = "PASSED"; - //if( result.hasMessage() ) - if( _stats.infoMessages.size() == 1 ) - messageLabel = "with message"; - if( _stats.infoMessages.size() > 1 ) - messageLabel = "with messages"; - break; - case ResultWas::ExpressionFailed: - if( result.isOk() ) { - colour = Colour::Success; - passOrFail = "FAILED - but was ok"; - } - else { - colour = Colour::Error; - passOrFail = "FAILED"; - } - if( _stats.infoMessages.size() == 1 ) - messageLabel = "with message"; - if( _stats.infoMessages.size() > 1 ) - messageLabel = "with messages"; - break; - case ResultWas::ThrewException: - colour = Colour::Error; - passOrFail = "FAILED"; - messageLabel = "due to unexpected exception with "; - if (_stats.infoMessages.size() == 1) - messageLabel += "message"; - if (_stats.infoMessages.size() > 1) - messageLabel += "messages"; - break; - case ResultWas::FatalErrorCondition: - colour = Colour::Error; - passOrFail = "FAILED"; - messageLabel = "due to a fatal error condition"; - break; - case ResultWas::DidntThrowException: - colour = Colour::Error; - passOrFail = "FAILED"; - messageLabel = "because no exception was thrown where one was expected"; - break; - case ResultWas::Info: - messageLabel = "info"; - break; - case ResultWas::Warning: - messageLabel = "warning"; - break; - case ResultWas::ExplicitFailure: - passOrFail = "FAILED"; - colour = Colour::Error; - if( _stats.infoMessages.size() == 1 ) - messageLabel = "explicitly with message"; - if( _stats.infoMessages.size() > 1 ) - messageLabel = "explicitly with messages"; - break; - // These cases are here to prevent compiler warnings - case ResultWas::Unknown: - case ResultWas::FailureBit: - case ResultWas::Exception: - passOrFail = "** internal error **"; - colour = Colour::Error; - break; - } - } - - void print() const { - printSourceInfo(); - if( stats.totals.assertions.total() > 0 ) { - if( result.isOk() ) - stream << '\n'; - printResultType(); - printOriginalExpression(); - printReconstructedExpression(); - } - else { - stream << '\n'; - } - printMessage(); - } - - private: - void printResultType() const { - if( !passOrFail.empty() ) { - Colour colourGuard( colour ); - stream << passOrFail << ":\n"; - } - } - void printOriginalExpression() const { - if( result.hasExpression() ) { - Colour colourGuard( Colour::OriginalExpression ); - stream << " "; - stream << result.getExpressionInMacro(); - stream << '\n'; - } - } - void printReconstructedExpression() const { - if( result.hasExpandedExpression() ) { - stream << "with expansion:\n"; - Colour colourGuard( Colour::ReconstructedExpression ); - stream << Text( result.getExpandedExpression(), TextAttributes().setIndent(2) ) << '\n'; - } - } - void printMessage() const { - if( !messageLabel.empty() ) - stream << messageLabel << ':' << '\n'; - for( std::vector::const_iterator it = messages.begin(), itEnd = messages.end(); - it != itEnd; - ++it ) { - // If this assertion is a warning ignore any INFO messages - if( printInfoMessages || it->type != ResultWas::Info ) - stream << Text( it->message, TextAttributes().setIndent(2) ) << '\n'; - } - } - void printSourceInfo() const { - Colour colourGuard( Colour::FileName ); - stream << result.getSourceInfo() << ": "; - } - - std::ostream& stream; - AssertionStats const& stats; - AssertionResult const& result; - Colour::Code colour; - std::string passOrFail; - std::string messageLabel; - std::string message; - std::vector messages; - bool printInfoMessages; - }; - - void lazyPrint() { - - if( !currentTestRunInfo.used ) - lazyPrintRunInfo(); - if( !currentGroupInfo.used ) - lazyPrintGroupInfo(); - - if( !m_headerPrinted ) { - printTestCaseAndSectionHeader(); - m_headerPrinted = true; - } - } - void lazyPrintRunInfo() { - stream << '\n' << getLineOfChars<'~'>() << '\n'; - Colour colour( Colour::SecondaryText ); - stream << currentTestRunInfo->name - << " is a Catch v" << libraryVersion() << " host application.\n" - << "Run with -? for options\n\n"; - - if( m_config->rngSeed() != 0 ) - stream << "Randomness seeded to: " << m_config->rngSeed() << "\n\n"; - - currentTestRunInfo.used = true; - } - void lazyPrintGroupInfo() { - if( !currentGroupInfo->name.empty() && currentGroupInfo->groupsCounts > 1 ) { - printClosedHeader( "Group: " + currentGroupInfo->name ); - currentGroupInfo.used = true; - } - } - void printTestCaseAndSectionHeader() { - assert( !m_sectionStack.empty() ); - printOpenHeader( currentTestCaseInfo->name ); - - if( m_sectionStack.size() > 1 ) { - Colour colourGuard( Colour::Headers ); - - std::vector::const_iterator - it = m_sectionStack.begin()+1, // Skip first section (test case) - itEnd = m_sectionStack.end(); - for( ; it != itEnd; ++it ) - printHeaderString( it->name, 2 ); - } - - SourceLineInfo lineInfo = m_sectionStack.back().lineInfo; - - if( !lineInfo.empty() ){ - stream << getLineOfChars<'-'>() << '\n'; - Colour colourGuard( Colour::FileName ); - stream << lineInfo << '\n'; - } - stream << getLineOfChars<'.'>() << '\n' << std::endl; - } - - void printClosedHeader( std::string const& _name ) { - printOpenHeader( _name ); - stream << getLineOfChars<'.'>() << '\n'; - } - void printOpenHeader( std::string const& _name ) { - stream << getLineOfChars<'-'>() << '\n'; - { - Colour colourGuard( Colour::Headers ); - printHeaderString( _name ); - } - } - - // if string has a : in first line will set indent to follow it on - // subsequent lines - void printHeaderString( std::string const& _string, std::size_t indent = 0 ) { - std::size_t i = _string.find( ": " ); - if( i != std::string::npos ) - i+=2; - else - i = 0; - stream << Text( _string, TextAttributes() - .setIndent( indent+i) - .setInitialIndent( indent ) ) << '\n'; - } - - struct SummaryColumn { - - SummaryColumn( std::string const& _label, Colour::Code _colour ) - : label( _label ), - colour( _colour ) - {} - SummaryColumn addRow( std::size_t count ) { - std::ostringstream oss; - oss << count; - std::string row = oss.str(); - for( std::vector::iterator it = rows.begin(); it != rows.end(); ++it ) { - while( it->size() < row.size() ) - *it = ' ' + *it; - while( it->size() > row.size() ) - row = ' ' + row; - } - rows.push_back( row ); - return *this; - } - - std::string label; - Colour::Code colour; - std::vector rows; - - }; - - void printTotals( Totals const& totals ) { - if( totals.testCases.total() == 0 ) { - stream << Colour( Colour::Warning ) << "No tests ran\n"; - } - else if( totals.assertions.total() > 0 && totals.testCases.allPassed() ) { - stream << Colour( Colour::ResultSuccess ) << "All tests passed"; - stream << " (" - << pluralise( totals.assertions.passed, "assertion" ) << " in " - << pluralise( totals.testCases.passed, "test case" ) << ')' - << '\n'; - } - else { - - std::vector columns; - columns.push_back( SummaryColumn( "", Colour::None ) - .addRow( totals.testCases.total() ) - .addRow( totals.assertions.total() ) ); - columns.push_back( SummaryColumn( "passed", Colour::Success ) - .addRow( totals.testCases.passed ) - .addRow( totals.assertions.passed ) ); - columns.push_back( SummaryColumn( "failed", Colour::ResultError ) - .addRow( totals.testCases.failed ) - .addRow( totals.assertions.failed ) ); - columns.push_back( SummaryColumn( "failed as expected", Colour::ResultExpectedFailure ) - .addRow( totals.testCases.failedButOk ) - .addRow( totals.assertions.failedButOk ) ); - - printSummaryRow( "test cases", columns, 0 ); - printSummaryRow( "assertions", columns, 1 ); - } - } - void printSummaryRow( std::string const& label, std::vector const& cols, std::size_t row ) { - for( std::vector::const_iterator it = cols.begin(); it != cols.end(); ++it ) { - std::string value = it->rows[row]; - if( it->label.empty() ) { - stream << label << ": "; - if( value != "0" ) - stream << value; - else - stream << Colour( Colour::Warning ) << "- none -"; - } - else if( value != "0" ) { - stream << Colour( Colour::LightGrey ) << " | "; - stream << Colour( it->colour ) - << value << ' ' << it->label; - } - } - stream << '\n'; - } - - static std::size_t makeRatio( std::size_t number, std::size_t total ) { - std::size_t ratio = total > 0 ? CATCH_CONFIG_CONSOLE_WIDTH * number/ total : 0; - return ( ratio == 0 && number > 0 ) ? 1 : ratio; - } - static std::size_t& findMax( std::size_t& i, std::size_t& j, std::size_t& k ) { - if( i > j && i > k ) - return i; - else if( j > k ) - return j; - else - return k; - } - - void printTotalsDivider( Totals const& totals ) { - if( totals.testCases.total() > 0 ) { - std::size_t failedRatio = makeRatio( totals.testCases.failed, totals.testCases.total() ); - std::size_t failedButOkRatio = makeRatio( totals.testCases.failedButOk, totals.testCases.total() ); - std::size_t passedRatio = makeRatio( totals.testCases.passed, totals.testCases.total() ); - while( failedRatio + failedButOkRatio + passedRatio < CATCH_CONFIG_CONSOLE_WIDTH-1 ) - findMax( failedRatio, failedButOkRatio, passedRatio )++; - while( failedRatio + failedButOkRatio + passedRatio > CATCH_CONFIG_CONSOLE_WIDTH-1 ) - findMax( failedRatio, failedButOkRatio, passedRatio )--; - - stream << Colour( Colour::Error ) << std::string( failedRatio, '=' ); - stream << Colour( Colour::ResultExpectedFailure ) << std::string( failedButOkRatio, '=' ); - if( totals.testCases.allPassed() ) - stream << Colour( Colour::ResultSuccess ) << std::string( passedRatio, '=' ); - else - stream << Colour( Colour::Success ) << std::string( passedRatio, '=' ); - } - else { - stream << Colour( Colour::Warning ) << std::string( CATCH_CONFIG_CONSOLE_WIDTH-1, '=' ); - } - stream << '\n'; - } - void printSummaryDivider() { - stream << getLineOfChars<'-'>() << '\n'; - } - - private: - bool m_headerPrinted; - }; - - INTERNAL_CATCH_REGISTER_REPORTER( "console", ConsoleReporter ) - -} // end namespace Catch - -// #included from: ../reporters/catch_reporter_compact.hpp -#define TWOBLUECUBES_CATCH_REPORTER_COMPACT_HPP_INCLUDED - -namespace Catch { - - struct CompactReporter : StreamingReporterBase { - - CompactReporter( ReporterConfig const& _config ) - : StreamingReporterBase( _config ) - {} - - virtual ~CompactReporter(); - - static std::string getDescription() { - return "Reports test results on a single line, suitable for IDEs"; - } - - virtual ReporterPreferences getPreferences() const { - ReporterPreferences prefs; - prefs.shouldRedirectStdOut = false; - return prefs; - } - - virtual void noMatchingTestCases( std::string const& spec ) { - stream << "No test cases matched '" << spec << '\'' << std::endl; - } - - virtual void assertionStarting( AssertionInfo const& ) {} - - virtual bool assertionEnded( AssertionStats const& _assertionStats ) { - AssertionResult const& result = _assertionStats.assertionResult; - - bool printInfoMessages = true; - - // Drop out if result was successful and we're not printing those - if( !m_config->includeSuccessfulResults() && result.isOk() ) { - if( result.getResultType() != ResultWas::Warning ) - return false; - printInfoMessages = false; - } - - AssertionPrinter printer( stream, _assertionStats, printInfoMessages ); - printer.print(); - - stream << std::endl; - return true; - } - - virtual void sectionEnded(SectionStats const& _sectionStats) CATCH_OVERRIDE { - if (m_config->showDurations() == ShowDurations::Always) { - stream << getFormattedDuration(_sectionStats.durationInSeconds) << " s: " << _sectionStats.sectionInfo.name << std::endl; - } - } - - virtual void testRunEnded( TestRunStats const& _testRunStats ) { - printTotals( _testRunStats.totals ); - stream << '\n' << std::endl; - StreamingReporterBase::testRunEnded( _testRunStats ); - } - - private: - class AssertionPrinter { - void operator= ( AssertionPrinter const& ); - public: - AssertionPrinter( std::ostream& _stream, AssertionStats const& _stats, bool _printInfoMessages ) - : stream( _stream ) - , stats( _stats ) - , result( _stats.assertionResult ) - , messages( _stats.infoMessages ) - , itMessage( _stats.infoMessages.begin() ) - , printInfoMessages( _printInfoMessages ) - {} - - void print() { - printSourceInfo(); - - itMessage = messages.begin(); - - switch( result.getResultType() ) { - case ResultWas::Ok: - printResultType( Colour::ResultSuccess, passedString() ); - printOriginalExpression(); - printReconstructedExpression(); - if ( ! result.hasExpression() ) - printRemainingMessages( Colour::None ); - else - printRemainingMessages(); - break; - case ResultWas::ExpressionFailed: - if( result.isOk() ) - printResultType( Colour::ResultSuccess, failedString() + std::string( " - but was ok" ) ); - else - printResultType( Colour::Error, failedString() ); - printOriginalExpression(); - printReconstructedExpression(); - printRemainingMessages(); - break; - case ResultWas::ThrewException: - printResultType( Colour::Error, failedString() ); - printIssue( "unexpected exception with message:" ); - printMessage(); - printExpressionWas(); - printRemainingMessages(); - break; - case ResultWas::FatalErrorCondition: - printResultType( Colour::Error, failedString() ); - printIssue( "fatal error condition with message:" ); - printMessage(); - printExpressionWas(); - printRemainingMessages(); - break; - case ResultWas::DidntThrowException: - printResultType( Colour::Error, failedString() ); - printIssue( "expected exception, got none" ); - printExpressionWas(); - printRemainingMessages(); - break; - case ResultWas::Info: - printResultType( Colour::None, "info" ); - printMessage(); - printRemainingMessages(); - break; - case ResultWas::Warning: - printResultType( Colour::None, "warning" ); - printMessage(); - printRemainingMessages(); - break; - case ResultWas::ExplicitFailure: - printResultType( Colour::Error, failedString() ); - printIssue( "explicitly" ); - printRemainingMessages( Colour::None ); - break; - // These cases are here to prevent compiler warnings - case ResultWas::Unknown: - case ResultWas::FailureBit: - case ResultWas::Exception: - printResultType( Colour::Error, "** internal error **" ); - break; - } - } - - private: - // Colour::LightGrey - - static Colour::Code dimColour() { return Colour::FileName; } - -#ifdef CATCH_PLATFORM_MAC - static const char* failedString() { return "FAILED"; } - static const char* passedString() { return "PASSED"; } -#else - static const char* failedString() { return "failed"; } - static const char* passedString() { return "passed"; } -#endif - - void printSourceInfo() const { - Colour colourGuard( Colour::FileName ); - stream << result.getSourceInfo() << ':'; - } - - void printResultType( Colour::Code colour, std::string const& passOrFail ) const { - if( !passOrFail.empty() ) { - { - Colour colourGuard( colour ); - stream << ' ' << passOrFail; - } - stream << ':'; - } - } - - void printIssue( std::string const& issue ) const { - stream << ' ' << issue; - } - - void printExpressionWas() { - if( result.hasExpression() ) { - stream << ';'; - { - Colour colour( dimColour() ); - stream << " expression was:"; - } - printOriginalExpression(); - } - } - - void printOriginalExpression() const { - if( result.hasExpression() ) { - stream << ' ' << result.getExpression(); - } - } - - void printReconstructedExpression() const { - if( result.hasExpandedExpression() ) { - { - Colour colour( dimColour() ); - stream << " for: "; - } - stream << result.getExpandedExpression(); - } - } - - void printMessage() { - if ( itMessage != messages.end() ) { - stream << " '" << itMessage->message << '\''; - ++itMessage; - } - } - - void printRemainingMessages( Colour::Code colour = dimColour() ) { - if ( itMessage == messages.end() ) - return; - - // using messages.end() directly yields compilation error: - std::vector::const_iterator itEnd = messages.end(); - const std::size_t N = static_cast( std::distance( itMessage, itEnd ) ); - - { - Colour colourGuard( colour ); - stream << " with " << pluralise( N, "message" ) << ':'; - } - - for(; itMessage != itEnd; ) { - // If this assertion is a warning ignore any INFO messages - if( printInfoMessages || itMessage->type != ResultWas::Info ) { - stream << " '" << itMessage->message << '\''; - if ( ++itMessage != itEnd ) { - Colour colourGuard( dimColour() ); - stream << " and"; - } - } - } - } - - private: - std::ostream& stream; - AssertionStats const& stats; - AssertionResult const& result; - std::vector messages; - std::vector::const_iterator itMessage; - bool printInfoMessages; - }; - - // 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 ) { - stream << "No tests ran."; - } - 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 { - Colour colour( Colour::ResultSuccess ); - stream << - "Passed " << bothOrAll( totals.testCases.passed ) - << pluralise( totals.testCases.passed, "test case" ) << - " with " << pluralise( totals.assertions.passed, "assertion" ) << '.'; - } - } - }; - - INTERNAL_CATCH_REGISTER_REPORTER( "compact", CompactReporter ) - -} // end namespace Catch - -namespace Catch { - // These are all here to avoid warnings about not having any out of line - // virtual methods - NonCopyable::~NonCopyable() {} - IShared::~IShared() {} - IStream::~IStream() CATCH_NOEXCEPT {} - FileStream::~FileStream() CATCH_NOEXCEPT {} - CoutStream::~CoutStream() CATCH_NOEXCEPT {} - DebugOutStream::~DebugOutStream() CATCH_NOEXCEPT {} - StreamBufBase::~StreamBufBase() CATCH_NOEXCEPT {} - IContext::~IContext() {} - IResultCapture::~IResultCapture() {} - ITestCase::~ITestCase() {} - ITestCaseRegistry::~ITestCaseRegistry() {} - IRegistryHub::~IRegistryHub() {} - IMutableRegistryHub::~IMutableRegistryHub() {} - IExceptionTranslator::~IExceptionTranslator() {} - IExceptionTranslatorRegistry::~IExceptionTranslatorRegistry() {} - IReporter::~IReporter() {} - IReporterFactory::~IReporterFactory() {} - IReporterRegistry::~IReporterRegistry() {} - IStreamingReporter::~IStreamingReporter() {} - AssertionStats::~AssertionStats() {} - SectionStats::~SectionStats() {} - TestCaseStats::~TestCaseStats() {} - TestGroupStats::~TestGroupStats() {} - TestRunStats::~TestRunStats() {} - CumulativeReporterBase::SectionNode::~SectionNode() {} - CumulativeReporterBase::~CumulativeReporterBase() {} - - StreamingReporterBase::~StreamingReporterBase() {} - ConsoleReporter::~ConsoleReporter() {} - CompactReporter::~CompactReporter() {} - IRunner::~IRunner() {} - IMutableContext::~IMutableContext() {} - IConfig::~IConfig() {} - XmlReporter::~XmlReporter() {} - JunitReporter::~JunitReporter() {} - TestRegistry::~TestRegistry() {} - FreeFunctionTestCase::~FreeFunctionTestCase() {} - IGeneratorInfo::~IGeneratorInfo() {} - IGeneratorsForTest::~IGeneratorsForTest() {} - WildcardPattern::~WildcardPattern() {} - TestSpec::Pattern::~Pattern() {} - TestSpec::NamePattern::~NamePattern() {} - TestSpec::TagPattern::~TagPattern() {} - TestSpec::ExcludedPattern::~ExcludedPattern() {} - Matchers::Impl::MatcherUntypedBase::~MatcherUntypedBase() {} - - void Config::dummy() {} - - namespace TestCaseTracking { - ITracker::~ITracker() {} - TrackerBase::~TrackerBase() {} - SectionTracker::~SectionTracker() {} - IndexTracker::~IndexTracker() {} - } -} - -#ifdef __clang__ -#pragma clang diagnostic pop -#endif - -#endif - -#ifdef CATCH_CONFIG_MAIN -// #included from: internal/catch_default_main.hpp -#define TWOBLUECUBES_CATCH_DEFAULT_MAIN_HPP_INCLUDED - -#ifndef __OBJC__ - -#if defined(WIN32) && defined(_UNICODE) && !defined(DO_NOT_USE_WMAIN) -// Standard C/C++ Win32 Unicode wmain entry point -extern "C" int wmain (int argc, wchar_t * argv[], wchar_t * []) { -#else -// Standard C/C++ main entry point -int main (int argc, char * argv[]) { -#endif - - int result = Catch::Session().run( argc, argv ); - return ( result < 0xff ? result : 0xff ); -} - -#else // __OBJC__ - -// Objective-C entry point -int main (int argc, char * const argv[]) { -#if !CATCH_ARC_ENABLED - NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init]; -#endif - - Catch::registerTestMethods(); - int result = Catch::Session().run( argc, (char* const*)argv ); - -#if !CATCH_ARC_ENABLED - [pool drain]; -#endif - - return ( result < 0xff ? result : 0xff ); -} - -#endif // __OBJC__ - -#endif - -#ifdef CLARA_CONFIG_MAIN_NOT_DEFINED -# undef CLARA_CONFIG_MAIN -#endif - -////// - -// If this config identifier is defined then all CATCH macros are prefixed with CATCH_ -#ifdef CATCH_CONFIG_PREFIX_ALL - -#if defined(CATCH_CONFIG_FAST_COMPILE) -#define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "CATCH_REQUIRE", Catch::ResultDisposition::Normal, expr ) -#define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "CATCH_REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) -#else -#define CATCH_REQUIRE( expr ) INTERNAL_CATCH_TEST( "CATCH_REQUIRE", Catch::ResultDisposition::Normal, expr ) -#define CATCH_REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( "CATCH_REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) -#endif - -#define CATCH_REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( "CATCH_REQUIRE_THROWS", Catch::ResultDisposition::Normal, "", expr ) -#define CATCH_REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CATCH_REQUIRE_THROWS_AS", exceptionType, Catch::ResultDisposition::Normal, expr ) -#define CATCH_REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CATCH_REQUIRE_THROWS_WITH", Catch::ResultDisposition::Normal, matcher, expr ) -#define CATCH_REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CATCH_REQUIRE_NOTHROW", Catch::ResultDisposition::Normal, expr ) - -#define CATCH_CHECK( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CATCH_CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK_FALSE", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, expr ) -#define CATCH_CHECKED_IF( expr ) INTERNAL_CATCH_IF( "CATCH_CHECKED_IF", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CATCH_CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( "CATCH_CHECKED_ELSE", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CATCH_CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( "CATCH_CHECK_NOFAIL", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, expr ) - -#define CATCH_CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( "CATCH_CHECK_THROWS", Catch::ResultDisposition::ContinueOnFailure, "", expr ) -#define CATCH_CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CATCH_CHECK_THROWS_AS", exceptionType, Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CATCH_CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CATCH_CHECK_THROWS_WITH", Catch::ResultDisposition::ContinueOnFailure, matcher, expr ) -#define CATCH_CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CATCH_CHECK_NOTHROW", Catch::ResultDisposition::ContinueOnFailure, expr ) - -#define CATCH_CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CATCH_CHECK_THAT", matcher, Catch::ResultDisposition::ContinueOnFailure, arg ) - -#if defined(CATCH_CONFIG_FAST_COMPILE) -#define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT_NO_TRY( "CATCH_REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) -#else -#define CATCH_REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CATCH_REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) -#endif - -#define CATCH_INFO( msg ) INTERNAL_CATCH_INFO( "CATCH_INFO", msg ) -#define CATCH_WARN( msg ) INTERNAL_CATCH_MSG( "CATCH_WARN", Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, msg ) -#define CATCH_SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( "CATCH_INFO", msg ) -#define CATCH_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CATCH_CAPTURE", #msg " := " << Catch::toString(msg) ) -#define CATCH_SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CATCH_CAPTURE", #msg " := " << Catch::toString(msg) ) - -#ifdef CATCH_CONFIG_VARIADIC_MACROS - #define CATCH_TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ ) - #define CATCH_TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ ) - #define CATCH_METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ ) - #define CATCH_REGISTER_TEST_CASE( Function, ... ) INTERNAL_CATCH_REGISTER_TESTCASE( Function, __VA_ARGS__ ) - #define CATCH_SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ ) - #define CATCH_FAIL( ... ) INTERNAL_CATCH_MSG( "CATCH_FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, __VA_ARGS__ ) - #define CATCH_FAIL_CHECK( ... ) INTERNAL_CATCH_MSG( "CATCH_FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) - #define CATCH_SUCCEED( ... ) INTERNAL_CATCH_MSG( "CATCH_SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) -#else - #define CATCH_TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description ) - #define CATCH_TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description ) - #define CATCH_METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description ) - #define CATCH_REGISTER_TEST_CASE( function, name, description ) INTERNAL_CATCH_REGISTER_TESTCASE( function, name, description ) - #define CATCH_SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description ) - #define CATCH_FAIL( msg ) INTERNAL_CATCH_MSG( "CATCH_FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, msg ) - #define CATCH_FAIL_CHECK( msg ) INTERNAL_CATCH_MSG( "CATCH_FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, msg ) - #define CATCH_SUCCEED( msg ) INTERNAL_CATCH_MSG( "CATCH_SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, msg ) -#endif -#define CATCH_ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" ) - -#define CATCH_REGISTER_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType ) -#define CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) - -#define CATCH_GENERATE( expr) INTERNAL_CATCH_GENERATE( expr ) - -// "BDD-style" convenience wrappers -#ifdef CATCH_CONFIG_VARIADIC_MACROS -#define CATCH_SCENARIO( ... ) CATCH_TEST_CASE( "Scenario: " __VA_ARGS__ ) -#define CATCH_SCENARIO_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, "Scenario: " __VA_ARGS__ ) -#else -#define CATCH_SCENARIO( name, tags ) CATCH_TEST_CASE( "Scenario: " name, tags ) -#define CATCH_SCENARIO_METHOD( className, name, tags ) INTERNAL_CATCH_TEST_CASE_METHOD( className, "Scenario: " name, tags ) -#endif -#define CATCH_GIVEN( desc ) CATCH_SECTION( std::string( "Given: ") + desc, "" ) -#define CATCH_WHEN( desc ) CATCH_SECTION( std::string( " When: ") + desc, "" ) -#define CATCH_AND_WHEN( desc ) CATCH_SECTION( std::string( " And: ") + desc, "" ) -#define CATCH_THEN( desc ) CATCH_SECTION( std::string( " Then: ") + desc, "" ) -#define CATCH_AND_THEN( desc ) CATCH_SECTION( std::string( " And: ") + desc, "" ) - -// If CATCH_CONFIG_PREFIX_ALL is not defined then the CATCH_ prefix is not required -#else - -#if defined(CATCH_CONFIG_FAST_COMPILE) -#define REQUIRE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "REQUIRE", Catch::ResultDisposition::Normal, expr ) -#define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST_NO_TRY( "REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) - -#else -#define REQUIRE( expr ) INTERNAL_CATCH_TEST( "REQUIRE", Catch::ResultDisposition::Normal, expr ) -#define REQUIRE_FALSE( expr ) INTERNAL_CATCH_TEST( "REQUIRE_FALSE", Catch::ResultDisposition::Normal | Catch::ResultDisposition::FalseTest, expr ) -#endif - -#define REQUIRE_THROWS( expr ) INTERNAL_CATCH_THROWS( "REQUIRE_THROWS", Catch::ResultDisposition::Normal, "", expr ) -#define REQUIRE_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "REQUIRE_THROWS_AS", exceptionType, Catch::ResultDisposition::Normal, expr ) -#define REQUIRE_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "REQUIRE_THROWS_WITH", Catch::ResultDisposition::Normal, matcher, expr ) -#define REQUIRE_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "REQUIRE_NOTHROW", Catch::ResultDisposition::Normal, expr ) - -#define CHECK( expr ) INTERNAL_CATCH_TEST( "CHECK", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CHECK_FALSE( expr ) INTERNAL_CATCH_TEST( "CHECK_FALSE", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::FalseTest, expr ) -#define CHECKED_IF( expr ) INTERNAL_CATCH_IF( "CHECKED_IF", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CHECKED_ELSE( expr ) INTERNAL_CATCH_ELSE( "CHECKED_ELSE", Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CHECK_NOFAIL( expr ) INTERNAL_CATCH_TEST( "CHECK_NOFAIL", Catch::ResultDisposition::ContinueOnFailure | Catch::ResultDisposition::SuppressFail, expr ) - -#define CHECK_THROWS( expr ) INTERNAL_CATCH_THROWS( "CHECK_THROWS", Catch::ResultDisposition::ContinueOnFailure, "", expr ) -#define CHECK_THROWS_AS( expr, exceptionType ) INTERNAL_CATCH_THROWS_AS( "CHECK_THROWS_AS", exceptionType, Catch::ResultDisposition::ContinueOnFailure, expr ) -#define CHECK_THROWS_WITH( expr, matcher ) INTERNAL_CATCH_THROWS( "CHECK_THROWS_WITH", Catch::ResultDisposition::ContinueOnFailure, matcher, expr ) -#define CHECK_NOTHROW( expr ) INTERNAL_CATCH_NO_THROW( "CHECK_NOTHROW", Catch::ResultDisposition::ContinueOnFailure, expr ) - -#define CHECK_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "CHECK_THAT", matcher, Catch::ResultDisposition::ContinueOnFailure, arg ) - -#if defined(CATCH_CONFIG_FAST_COMPILE) -#define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT_NO_TRY( "REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) -#else -#define REQUIRE_THAT( arg, matcher ) INTERNAL_CHECK_THAT( "REQUIRE_THAT", matcher, Catch::ResultDisposition::Normal, arg ) -#endif - -#define INFO( msg ) INTERNAL_CATCH_INFO( "INFO", msg ) -#define WARN( msg ) INTERNAL_CATCH_MSG( "WARN", Catch::ResultWas::Warning, Catch::ResultDisposition::ContinueOnFailure, msg ) -#define SCOPED_INFO( msg ) INTERNAL_CATCH_INFO( "INFO", msg ) -#define CAPTURE( msg ) INTERNAL_CATCH_INFO( "CAPTURE", #msg " := " << Catch::toString(msg) ) -#define SCOPED_CAPTURE( msg ) INTERNAL_CATCH_INFO( "CAPTURE", #msg " := " << Catch::toString(msg) ) - -#ifdef CATCH_CONFIG_VARIADIC_MACROS -#define TEST_CASE( ... ) INTERNAL_CATCH_TESTCASE( __VA_ARGS__ ) -#define TEST_CASE_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, __VA_ARGS__ ) -#define METHOD_AS_TEST_CASE( method, ... ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, __VA_ARGS__ ) -#define REGISTER_TEST_CASE( Function, ... ) INTERNAL_CATCH_REGISTER_TESTCASE( Function, __VA_ARGS__ ) -#define SECTION( ... ) INTERNAL_CATCH_SECTION( __VA_ARGS__ ) -#define FAIL( ... ) INTERNAL_CATCH_MSG( "FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, __VA_ARGS__ ) -#define FAIL_CHECK( ... ) INTERNAL_CATCH_MSG( "FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) -#define SUCCEED( ... ) INTERNAL_CATCH_MSG( "SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, __VA_ARGS__ ) -#else -#define TEST_CASE( name, description ) INTERNAL_CATCH_TESTCASE( name, description ) - #define TEST_CASE_METHOD( className, name, description ) INTERNAL_CATCH_TEST_CASE_METHOD( className, name, description ) - #define METHOD_AS_TEST_CASE( method, name, description ) INTERNAL_CATCH_METHOD_AS_TEST_CASE( method, name, description ) - #define REGISTER_TEST_CASE( method, name, description ) INTERNAL_CATCH_REGISTER_TESTCASE( method, name, description ) - #define SECTION( name, description ) INTERNAL_CATCH_SECTION( name, description ) - #define FAIL( msg ) INTERNAL_CATCH_MSG( "FAIL", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::Normal, msg ) - #define FAIL_CHECK( msg ) INTERNAL_CATCH_MSG( "FAIL_CHECK", Catch::ResultWas::ExplicitFailure, Catch::ResultDisposition::ContinueOnFailure, msg ) - #define SUCCEED( msg ) INTERNAL_CATCH_MSG( "SUCCEED", Catch::ResultWas::Ok, Catch::ResultDisposition::ContinueOnFailure, msg ) -#endif -#define ANON_TEST_CASE() INTERNAL_CATCH_TESTCASE( "", "" ) - -#define REGISTER_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_REPORTER( name, reporterType ) -#define REGISTER_LEGACY_REPORTER( name, reporterType ) INTERNAL_CATCH_REGISTER_LEGACY_REPORTER( name, reporterType ) - -#define GENERATE( expr) INTERNAL_CATCH_GENERATE( expr ) - -#endif - -#define CATCH_TRANSLATE_EXCEPTION( signature ) INTERNAL_CATCH_TRANSLATE_EXCEPTION( signature ) - -// "BDD-style" convenience wrappers -#ifdef CATCH_CONFIG_VARIADIC_MACROS -#define SCENARIO( ... ) TEST_CASE( "Scenario: " __VA_ARGS__ ) -#define SCENARIO_METHOD( className, ... ) INTERNAL_CATCH_TEST_CASE_METHOD( className, "Scenario: " __VA_ARGS__ ) -#else -#define SCENARIO( name, tags ) TEST_CASE( "Scenario: " name, tags ) -#define SCENARIO_METHOD( className, name, tags ) INTERNAL_CATCH_TEST_CASE_METHOD( className, "Scenario: " name, tags ) -#endif -#define GIVEN( desc ) SECTION( std::string(" Given: ") + desc, "" ) -#define WHEN( desc ) SECTION( std::string(" When: ") + desc, "" ) -#define AND_WHEN( desc ) SECTION( std::string("And when: ") + desc, "" ) -#define THEN( desc ) SECTION( std::string(" Then: ") + desc, "" ) -#define AND_THEN( desc ) SECTION( std::string(" And: ") + desc, "" ) - -using Catch::Detail::Approx; - -// #included from: internal/catch_reenable_warnings.h - -#define TWOBLUECUBES_CATCH_REENABLE_WARNINGS_H_INCLUDED - -#ifdef __clang__ -# ifdef __ICC // icpc defines the __clang__ macro -# pragma warning(pop) -# else -# pragma clang diagnostic pop -# endif -#elif defined __GNUC__ -# pragma GCC diagnostic pop -#endif - -#endif // TWOBLUECUBES_SINGLE_INCLUDE_CATCH_HPP_INCLUDED - diff --git a/test/thirdparty/doctest/LICENSE.txt b/test/thirdparty/doctest/LICENSE.txt new file mode 100644 index 00000000..62cdd948 --- /dev/null +++ b/test/thirdparty/doctest/LICENSE.txt @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016-2018 Viktor Kirilov + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/test/thirdparty/doctest/doctest.h b/test/thirdparty/doctest/doctest.h new file mode 100644 index 00000000..eda93fc7 --- /dev/null +++ b/test/thirdparty/doctest/doctest.h @@ -0,0 +1,5269 @@ +// ====================================================================== +// == DO NOT MODIFY THIS FILE BY HAND - IT IS AUTO GENERATED BY CMAKE! == +// ====================================================================== +// +// doctest.h - the lightest feature-rich C++ single-header testing framework for unit tests and TDD +// +// Copyright (c) 2016-2018 Viktor Kirilov +// +// Distributed under the MIT Software License +// See accompanying file LICENSE.txt or copy at +// https://opensource.org/licenses/MIT +// +// The documentation can be found at the library's page: +// https://github.com/onqtam/doctest/blob/master/doc/markdown/readme.md +// +// ================================================================================================= +// ================================================================================================= +// ================================================================================================= +// +// The library is heavily influenced by Catch - https://github.com/philsquared/Catch +// which uses the Boost Software License - Version 1.0 +// see here - https://github.com/philsquared/Catch/blob/master/LICENSE.txt +// +// The concept of subcases (sections in Catch) and expression decomposition are from there. +// Some parts of the code are taken directly: +// - stringification - the detection of "ostream& operator<<(ostream&, const T&)" and StringMaker<> +// - the Approx() helper class for floating point comparison +// - colors in the console +// - breaking into a debugger +// - signal / SEH handling +// - timer +// +// The expression decomposing templates are taken from lest - https://github.com/martinmoene/lest +// which uses the Boost Software License - Version 1.0 +// see here - https://github.com/martinmoene/lest/blob/master/LICENSE.txt +// +// ================================================================================================= +// ================================================================================================= +// ================================================================================================= + +#ifndef DOCTEST_LIBRARY_INCLUDED +#define DOCTEST_LIBRARY_INCLUDED + +// ================================================================================================= +// == VERSION ====================================================================================== +// ================================================================================================= + +#define DOCTEST_VERSION_MAJOR 2 +#define DOCTEST_VERSION_MINOR 2 +#define DOCTEST_VERSION_PATCH 1 +#define DOCTEST_VERSION_STR "2.2.1" + +#define DOCTEST_VERSION \ + (DOCTEST_VERSION_MAJOR * 10000 + DOCTEST_VERSION_MINOR * 100 + DOCTEST_VERSION_PATCH) + +// ================================================================================================= +// == COMPILER VERSION ============================================================================= +// ================================================================================================= + +// ideas for the version stuff are taken from here: https://github.com/cxxstuff/cxx_detect + +#define DOCTEST_COMPILER(MAJOR, MINOR, PATCH) ((MAJOR)*10000000 + (MINOR)*100000 + (PATCH)) + +// GCC/Clang and GCC/MSVC are mutually exclusive, but Clang/MSVC are not because of clang-cl... +#if defined(_MSC_VER) && defined(_MSC_FULL_VER) +#if _MSC_VER == _MSC_FULL_VER / 10000 +#define DOCTEST_MSVC DOCTEST_COMPILER(_MSC_VER / 100, _MSC_VER % 100, _MSC_FULL_VER % 10000) +#else // MSVC +#define DOCTEST_MSVC \ + DOCTEST_COMPILER(_MSC_VER / 100, (_MSC_FULL_VER / 100000) % 100, _MSC_FULL_VER % 100000) +#endif // MSVC +#endif // MSVC +#if defined(__clang__) && defined(__clang_minor__) +#define DOCTEST_CLANG DOCTEST_COMPILER(__clang_major__, __clang_minor__, __clang_patchlevel__) +#elif defined(__GNUC__) && defined(__GNUC_MINOR__) && defined(__GNUC_PATCHLEVEL__) && \ + !defined(__INTEL_COMPILER) +#define DOCTEST_GCC DOCTEST_COMPILER(__GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__) +#endif // GCC + +#ifndef DOCTEST_MSVC +#define DOCTEST_MSVC 0 +#endif // DOCTEST_MSVC +#ifndef DOCTEST_CLANG +#define DOCTEST_CLANG 0 +#endif // DOCTEST_CLANG +#ifndef DOCTEST_GCC +#define DOCTEST_GCC 0 +#endif // DOCTEST_GCC + +// ================================================================================================= +// == COMPILER WARNINGS HELPERS ==================================================================== +// ================================================================================================= + +#if DOCTEST_CLANG +#define DOCTEST_PRAGMA_TO_STR(x) _Pragma(#x) +#define DOCTEST_CLANG_SUPPRESS_WARNING_PUSH _Pragma("clang diagnostic push") +#define DOCTEST_CLANG_SUPPRESS_WARNING(w) DOCTEST_PRAGMA_TO_STR(clang diagnostic ignored w) +#define DOCTEST_CLANG_SUPPRESS_WARNING_POP _Pragma("clang diagnostic pop") +#define DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH(w) \ + DOCTEST_CLANG_SUPPRESS_WARNING_PUSH DOCTEST_CLANG_SUPPRESS_WARNING(w) +#else // DOCTEST_CLANG +#define DOCTEST_CLANG_SUPPRESS_WARNING_PUSH +#define DOCTEST_CLANG_SUPPRESS_WARNING(w) +#define DOCTEST_CLANG_SUPPRESS_WARNING_POP +#define DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH(w) +#endif // DOCTEST_CLANG + +#if DOCTEST_GCC +#define DOCTEST_PRAGMA_TO_STR(x) _Pragma(#x) +#define DOCTEST_GCC_SUPPRESS_WARNING_PUSH _Pragma("GCC diagnostic push") +#define DOCTEST_GCC_SUPPRESS_WARNING(w) DOCTEST_PRAGMA_TO_STR(GCC diagnostic ignored w) +#define DOCTEST_GCC_SUPPRESS_WARNING_POP _Pragma("GCC diagnostic pop") +#define DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH(w) \ + DOCTEST_GCC_SUPPRESS_WARNING_PUSH DOCTEST_GCC_SUPPRESS_WARNING(w) +#else // DOCTEST_GCC +#define DOCTEST_GCC_SUPPRESS_WARNING_PUSH +#define DOCTEST_GCC_SUPPRESS_WARNING(w) +#define DOCTEST_GCC_SUPPRESS_WARNING_POP +#define DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH(w) +#endif // DOCTEST_GCC + +#if DOCTEST_MSVC +#define DOCTEST_MSVC_SUPPRESS_WARNING_PUSH __pragma(warning(push)) +#define DOCTEST_MSVC_SUPPRESS_WARNING(w) __pragma(warning(disable : w)) +#define DOCTEST_MSVC_SUPPRESS_WARNING_POP __pragma(warning(pop)) +#define DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(w) \ + DOCTEST_MSVC_SUPPRESS_WARNING_PUSH DOCTEST_MSVC_SUPPRESS_WARNING(w) +#else // DOCTEST_MSVC +#define DOCTEST_MSVC_SUPPRESS_WARNING_PUSH +#define DOCTEST_MSVC_SUPPRESS_WARNING(w) +#define DOCTEST_MSVC_SUPPRESS_WARNING_POP +#define DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(w) +#endif // DOCTEST_MSVC + +// ================================================================================================= +// == COMPILER WARNINGS ============================================================================ +// ================================================================================================= + +DOCTEST_CLANG_SUPPRESS_WARNING_PUSH +DOCTEST_CLANG_SUPPRESS_WARNING("-Wunknown-pragmas") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wnon-virtual-dtor") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wweak-vtables") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wpadded") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wdeprecated") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-prototypes") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-local-typedef") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat-pedantic") + +DOCTEST_GCC_SUPPRESS_WARNING_PUSH +DOCTEST_GCC_SUPPRESS_WARNING("-Wunknown-pragmas") +DOCTEST_GCC_SUPPRESS_WARNING("-Wpragmas") +DOCTEST_GCC_SUPPRESS_WARNING("-Weffc++") +DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-overflow") +DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-aliasing") +DOCTEST_GCC_SUPPRESS_WARNING("-Wctor-dtor-privacy") +DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-declarations") +DOCTEST_GCC_SUPPRESS_WARNING("-Wnon-virtual-dtor") +DOCTEST_GCC_SUPPRESS_WARNING("-Winline") +DOCTEST_GCC_SUPPRESS_WARNING("-Wunused-local-typedefs") +DOCTEST_GCC_SUPPRESS_WARNING("-Wuseless-cast") + +DOCTEST_MSVC_SUPPRESS_WARNING_PUSH +DOCTEST_MSVC_SUPPRESS_WARNING(4616) // invalid compiler warning +DOCTEST_MSVC_SUPPRESS_WARNING(4619) // invalid compiler warning +DOCTEST_MSVC_SUPPRESS_WARNING(4996) // The compiler encountered a deprecated declaration +DOCTEST_MSVC_SUPPRESS_WARNING(4706) // assignment within conditional expression +DOCTEST_MSVC_SUPPRESS_WARNING(4512) // 'class' : assignment operator could not be generated +DOCTEST_MSVC_SUPPRESS_WARNING(4127) // conditional expression is constant +DOCTEST_MSVC_SUPPRESS_WARNING(4820) // padding +DOCTEST_MSVC_SUPPRESS_WARNING(4625) // copy constructor was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(4626) // assignment operator was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(5027) // move assignment operator was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(5026) // move constructor was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(4623) // default constructor was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(4640) // construction of local static object is not thread-safe +// static analysis +DOCTEST_MSVC_SUPPRESS_WARNING(26439) // This kind of function may not throw. Declare it 'noexcept' +DOCTEST_MSVC_SUPPRESS_WARNING(26495) // Always initialize a member variable +DOCTEST_MSVC_SUPPRESS_WARNING(26451) // Arithmetic overflow ... +DOCTEST_MSVC_SUPPRESS_WARNING(26444) // Avoid unnamed objects with custom construction and dtr... + +// 4548 - expression before comma has no effect; expected expression with side - effect +// 4265 - class has virtual functions, but destructor is not virtual +// 4986 - exception specification does not match previous declaration +// 4350 - behavior change: 'member1' called instead of 'member2' +// 4668 - 'x' is not defined as a preprocessor macro, replacing with '0' for '#if/#elif' +// 4365 - conversion from 'int' to 'unsigned long', signed/unsigned mismatch +// 4774 - format string expected in argument 'x' is not a string literal +// 4820 - padding in structs + +// only 4 should be disabled globally: +// - 4514 # unreferenced inline function has been removed +// - 4571 # SEH related +// - 4710 # function not inlined +// - 4711 # function 'x' selected for automatic inline expansion + +#define DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_BEGIN \ + DOCTEST_MSVC_SUPPRESS_WARNING_PUSH \ + DOCTEST_MSVC_SUPPRESS_WARNING(4548) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4265) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4986) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4350) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4668) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4365) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4774) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4820) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4625) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4626) \ + DOCTEST_MSVC_SUPPRESS_WARNING(5027) \ + DOCTEST_MSVC_SUPPRESS_WARNING(5026) \ + DOCTEST_MSVC_SUPPRESS_WARNING(4623) \ + DOCTEST_MSVC_SUPPRESS_WARNING(5039) + +#define DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_END DOCTEST_MSVC_SUPPRESS_WARNING_POP + +// ================================================================================================= +// == FEATURE DETECTION ============================================================================ +// ================================================================================================= + +// general compiler feature support table: https://en.cppreference.com/w/cpp/compiler_support +// MSVC C++11 feature support table: https://msdn.microsoft.com/en-us/library/hh567368.aspx +// GCC C++11 feature support table: https://gcc.gnu.org/projects/cxx-status.html +// MSVC version table: +// MSVC++ 15.0 _MSC_VER == 1910 (Visual Studio 2017) +// MSVC++ 14.0 _MSC_VER == 1900 (Visual Studio 2015) +// MSVC++ 12.0 _MSC_VER == 1800 (Visual Studio 2013) +// MSVC++ 11.0 _MSC_VER == 1700 (Visual Studio 2012) +// MSVC++ 10.0 _MSC_VER == 1600 (Visual Studio 2010) +// MSVC++ 9.0 _MSC_VER == 1500 (Visual Studio 2008) +// MSVC++ 8.0 _MSC_VER == 1400 (Visual Studio 2005) + +#if DOCTEST_MSVC && !defined(DOCTEST_CONFIG_WINDOWS_SEH) +#define DOCTEST_CONFIG_WINDOWS_SEH +#endif // MSVC +#if defined(DOCTEST_CONFIG_NO_WINDOWS_SEH) && defined(DOCTEST_CONFIG_WINDOWS_SEH) +#undef DOCTEST_CONFIG_WINDOWS_SEH +#endif // DOCTEST_CONFIG_NO_WINDOWS_SEH + +#if !defined(_WIN32) && !defined(__QNX__) && !defined(DOCTEST_CONFIG_POSIX_SIGNALS) +#define DOCTEST_CONFIG_POSIX_SIGNALS +#endif // _WIN32 +#if defined(DOCTEST_CONFIG_NO_POSIX_SIGNALS) && defined(DOCTEST_CONFIG_POSIX_SIGNALS) +#undef DOCTEST_CONFIG_POSIX_SIGNALS +#endif // DOCTEST_CONFIG_NO_POSIX_SIGNALS + +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS +#if(DOCTEST_GCC || (DOCTEST_CLANG && !DOCTEST_MSVC)) && !defined(__EXCEPTIONS) +#define DOCTEST_CONFIG_NO_EXCEPTIONS +#endif // clang and gcc +#if DOCTEST_MSVC && (defined(_HAS_EXCEPTIONS) && _HAS_EXCEPTIONS == 0) +#define DOCTEST_CONFIG_NO_EXCEPTIONS +#endif // MSVC +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + +#ifdef DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS +#define DOCTEST_CONFIG_NO_EXCEPTIONS +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS + +#if defined(DOCTEST_CONFIG_NO_EXCEPTIONS) && !defined(DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS) +#define DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS && !DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS + +#if defined(DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN) && !defined(DOCTEST_CONFIG_IMPLEMENT) +#define DOCTEST_CONFIG_IMPLEMENT +#endif // DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN + +#if defined(_WIN32) || defined(__CYGWIN__) +#if DOCTEST_MSVC +#define DOCTEST_SYMBOL_EXPORT __declspec(dllexport) +#define DOCTEST_SYMBOL_IMPORT __declspec(dllimport) +#else // MSVC +#define DOCTEST_SYMBOL_EXPORT __attribute__((dllexport)) +#define DOCTEST_SYMBOL_IMPORT __attribute__((dllimport)) +#endif // MSVC +#else // _WIN32 +#define DOCTEST_SYMBOL_EXPORT __attribute__((visibility("default"))) +#define DOCTEST_SYMBOL_IMPORT +#endif // _WIN32 + +#ifdef DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL +#ifdef DOCTEST_CONFIG_IMPLEMENT +#define DOCTEST_INTERFACE DOCTEST_SYMBOL_EXPORT +#else // DOCTEST_CONFIG_IMPLEMENT +#define DOCTEST_INTERFACE DOCTEST_SYMBOL_IMPORT +#endif // DOCTEST_CONFIG_IMPLEMENT +#else // DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL +#define DOCTEST_INTERFACE +#endif // DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL + +#if DOCTEST_MSVC +#define DOCTEST_NOINLINE __declspec(noinline) +#define DOCTEST_UNUSED +#define DOCTEST_ALIGNMENT(x) +#else // MSVC +#define DOCTEST_NOINLINE __attribute__((noinline)) +#define DOCTEST_UNUSED __attribute__((unused)) +#define DOCTEST_ALIGNMENT(x) __attribute__((aligned(x))) +#endif // MSVC + +#ifndef DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK +#define DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK 5 +#endif // DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK + +// ================================================================================================= +// == FEATURE DETECTION END ======================================================================== +// ================================================================================================= + +// internal macros for string concatenation and anonymous variable name generation +#define DOCTEST_CAT_IMPL(s1, s2) s1##s2 +#define DOCTEST_CAT(s1, s2) DOCTEST_CAT_IMPL(s1, s2) +#ifdef __COUNTER__ // not standard and may be missing for some compilers +#define DOCTEST_ANONYMOUS(x) DOCTEST_CAT(x, __COUNTER__) +#else // __COUNTER__ +#define DOCTEST_ANONYMOUS(x) DOCTEST_CAT(x, __LINE__) +#endif // __COUNTER__ + +#ifndef DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE +#define DOCTEST_REF_WRAP(x) x& +#else // DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE +#define DOCTEST_REF_WRAP(x) x +#endif // DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE + +// not using __APPLE__ because... this is how Catch does it +#if defined(__MAC_OS_X_VERSION_MIN_REQUIRED) +#define DOCTEST_PLATFORM_MAC +#elif defined(__IPHONE_OS_VERSION_MIN_REQUIRED) +#define DOCTEST_PLATFORM_IPHONE +#elif defined(_WIN32) +#define DOCTEST_PLATFORM_WINDOWS +#else // DOCTEST_PLATFORM +#define DOCTEST_PLATFORM_LINUX +#endif // DOCTEST_PLATFORM + +// clang-format off +#define DOCTEST_DELETE_COPIES(type) type(const type&) = delete; type& operator=(const type&) = delete +#define DOCTEST_DECLARE_COPIES(type) type(const type&); type& operator=(const type&) +#define DOCTEST_DEFINE_COPIES(type) type::type(const type&) = default; type& type::operator=(const type&) = default +#define DOCTEST_DECLARE_DEFAULTS(type) type(); ~type() +#define DOCTEST_DEFINE_DEFAULTS(type) type::type() = default; type::~type() = default +// clang-format on + +#define DOCTEST_GLOBAL_NO_WARNINGS(var) \ + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wglobal-constructors") \ + static int var DOCTEST_UNUSED // NOLINT(fuchsia-statically-constructed-objects,cert-err58-cpp) +#define DOCTEST_GLOBAL_NO_WARNINGS_END() DOCTEST_CLANG_SUPPRESS_WARNING_POP + +// should probably take a look at https://github.com/scottt/debugbreak +#ifdef DOCTEST_PLATFORM_MAC +#define DOCTEST_BREAK_INTO_DEBUGGER() __asm__("int $3\n" : :) +#elif DOCTEST_MSVC +#define DOCTEST_BREAK_INTO_DEBUGGER() __debugbreak() +#elif defined(__MINGW32__) +extern "C" __declspec(dllimport) void __stdcall DebugBreak(); +#define DOCTEST_BREAK_INTO_DEBUGGER() ::DebugBreak() +#else // linux +#define DOCTEST_BREAK_INTO_DEBUGGER() ((void)0) +#endif // linux + +#if DOCTEST_CLANG +// to detect if libc++ is being used with clang (the _LIBCPP_VERSION identifier) +#include +#endif // clang + +// Forward declaring 'X' in namespace std is not permitted by the C++ Standard. +DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4643) + +#if defined(_LIBCPP_VERSION) || defined(DOCTEST_CONFIG_USE_IOSFWD) +// not forward declaring ostream for libc++ because I had some problems (inline namespaces vs c++98) +// so the header is used - also it is very light and doesn't drag a ton of stuff +#include +#else // _LIBCPP_VERSION +namespace std { +template +struct char_traits; +template <> +struct char_traits; +template +class basic_ostream; +typedef basic_ostream > ostream; +} // namespace std +#endif // _LIBCPP_VERSION || DOCTEST_CONFIG_USE_IOSFWD + +#ifdef _LIBCPP_VERSION +#include +#else // _LIBCPP_VERSION +namespace std { +typedef decltype(nullptr) nullptr_t; +} +#endif // _LIBCPP_VERSION + +DOCTEST_MSVC_SUPPRESS_WARNING_POP + +#ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS +#include +#endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + +namespace doctest { + +DOCTEST_INTERFACE extern bool is_running_in_test; + +// A 24 byte string class (can be as small as 17 for x64 and 13 for x86) that can hold strings with length +// of up to 23 chars on the stack before going on the heap - the last byte of the buffer is used for: +// - "is small" bit - the highest bit - if "0" then it is small - otherwise its "1" (128) +// - if small - capacity left before going on the heap - using the lowest 5 bits +// - if small - 2 bits are left unused - the second and third highest ones +// - if small - acts as a null terminator if strlen() is 23 (24 including the null terminator) +// and the "is small" bit remains "0" ("as well as the capacity left") so its OK +// Idea taken from this lecture about the string implementation of facebook/folly - fbstring +// https://www.youtube.com/watch?v=kPR8h4-qZdk +// TODO: +// - optimizations - like not deleting memory unnecessarily in operator= and etc. +// - resize/reserve/clear +// - substr +// - replace +// - back/front +// - iterator stuff +// - find & friends +// - push_back/pop_back +// - assign/insert/erase +// - relational operators as free functions - taking const char* as one of the params +class DOCTEST_INTERFACE String +{ + static const unsigned len = 24; //!OCLINT avoid private static members + static const unsigned last = len - 1; //!OCLINT avoid private static members + + struct view // len should be more than sizeof(view) - because of the final byte for flags + { + char* ptr; + unsigned size; + unsigned capacity; + }; + + union + { + char buf[len]; + view data; + }; + + bool isOnStack() const { return (buf[last] & 128) == 0; } + void setOnHeap(); + void setLast(unsigned in = last); + + void copy(const String& other); + +public: + String(); + ~String(); + + String(const char* in); + String(const char* in, unsigned in_size); + + String(const String& other); + String& operator=(const String& other); + + String& operator+=(const String& other); + String operator+(const String& other) const; + + String(String&& other); + String& operator=(String&& other); + + char operator[](unsigned i) const; + char& operator[](unsigned i); + + // the only functions I'm willing to leave in the interface - available for inlining + const char* c_str() const { return const_cast(this)->c_str(); } // NOLINT + char* c_str() { + if(isOnStack()) + return reinterpret_cast(buf); + return data.ptr; + } + + unsigned size() const; + unsigned capacity() const; + + int compare(const char* other, bool no_case = false) const; + int compare(const String& other, bool no_case = false) const; +}; + +DOCTEST_INTERFACE bool operator==(const String& lhs, const String& rhs); +DOCTEST_INTERFACE bool operator!=(const String& lhs, const String& rhs); +DOCTEST_INTERFACE bool operator<(const String& lhs, const String& rhs); +DOCTEST_INTERFACE bool operator>(const String& lhs, const String& rhs); +DOCTEST_INTERFACE bool operator<=(const String& lhs, const String& rhs); +DOCTEST_INTERFACE bool operator>=(const String& lhs, const String& rhs); + +DOCTEST_INTERFACE std::ostream& operator<<(std::ostream& s, const String& in); + +namespace Color { + enum Enum + { + None = 0, + White, + Red, + Green, + Blue, + Cyan, + Yellow, + Grey, + + Bright = 0x10, + + BrightRed = Bright | Red, + BrightGreen = Bright | Green, + LightGrey = Bright | Grey, + BrightWhite = Bright | White + }; + + DOCTEST_INTERFACE std::ostream& operator<<(std::ostream& s, Color::Enum code); +} // namespace Color + +namespace assertType { + enum Enum + { + // macro traits + + is_warn = 1, + is_check = 2 * is_warn, + is_require = 2 * is_check, + + is_normal = 2 * is_require, + is_throws = 2 * is_normal, + is_throws_as = 2 * is_throws, + is_throws_with = 2 * is_throws_as, + is_nothrow = 2 * is_throws_with, + + is_false = 2 * is_nothrow, + is_unary = 2 * is_false, // not checked anywhere - used just to distinguish the types + + is_eq = 2 * is_unary, + is_ne = 2 * is_eq, + + is_lt = 2 * is_ne, + is_gt = 2 * is_lt, + + is_ge = 2 * is_gt, + is_le = 2 * is_ge, + + // macro types + + DT_WARN = is_normal | is_warn, + DT_CHECK = is_normal | is_check, + DT_REQUIRE = is_normal | is_require, + + DT_WARN_FALSE = is_normal | is_false | is_warn, + DT_CHECK_FALSE = is_normal | is_false | is_check, + DT_REQUIRE_FALSE = is_normal | is_false | is_require, + + DT_WARN_THROWS = is_throws | is_warn, + DT_CHECK_THROWS = is_throws | is_check, + DT_REQUIRE_THROWS = is_throws | is_require, + + DT_WARN_THROWS_AS = is_throws_as | is_warn, + DT_CHECK_THROWS_AS = is_throws_as | is_check, + DT_REQUIRE_THROWS_AS = is_throws_as | is_require, + + DT_WARN_THROWS_WITH = is_throws_with | is_warn, + DT_CHECK_THROWS_WITH = is_throws_with | is_check, + DT_REQUIRE_THROWS_WITH = is_throws_with | is_require, + + DT_WARN_NOTHROW = is_nothrow | is_warn, + DT_CHECK_NOTHROW = is_nothrow | is_check, + DT_REQUIRE_NOTHROW = is_nothrow | is_require, + + DT_WARN_EQ = is_normal | is_eq | is_warn, + DT_CHECK_EQ = is_normal | is_eq | is_check, + DT_REQUIRE_EQ = is_normal | is_eq | is_require, + + DT_WARN_NE = is_normal | is_ne | is_warn, + DT_CHECK_NE = is_normal | is_ne | is_check, + DT_REQUIRE_NE = is_normal | is_ne | is_require, + + DT_WARN_GT = is_normal | is_gt | is_warn, + DT_CHECK_GT = is_normal | is_gt | is_check, + DT_REQUIRE_GT = is_normal | is_gt | is_require, + + DT_WARN_LT = is_normal | is_lt | is_warn, + DT_CHECK_LT = is_normal | is_lt | is_check, + DT_REQUIRE_LT = is_normal | is_lt | is_require, + + DT_WARN_GE = is_normal | is_ge | is_warn, + DT_CHECK_GE = is_normal | is_ge | is_check, + DT_REQUIRE_GE = is_normal | is_ge | is_require, + + DT_WARN_LE = is_normal | is_le | is_warn, + DT_CHECK_LE = is_normal | is_le | is_check, + DT_REQUIRE_LE = is_normal | is_le | is_require, + + DT_WARN_UNARY = is_normal | is_unary | is_warn, + DT_CHECK_UNARY = is_normal | is_unary | is_check, + DT_REQUIRE_UNARY = is_normal | is_unary | is_require, + + DT_WARN_UNARY_FALSE = is_normal | is_false | is_unary | is_warn, + DT_CHECK_UNARY_FALSE = is_normal | is_false | is_unary | is_check, + DT_REQUIRE_UNARY_FALSE = is_normal | is_false | is_unary | is_require, + }; +} // namespace assertType + +DOCTEST_INTERFACE const char* assertString(assertType::Enum at); +DOCTEST_INTERFACE const char* failureString(assertType::Enum at); +DOCTEST_INTERFACE const char* removePathFromFilename(const char* file); + +struct DOCTEST_INTERFACE TestCaseData +{ + const char* m_file; // the file in which the test was registered + unsigned m_line; // the line where the test was registered + const char* m_name; // name of the test case + const char* m_test_suite; // the test suite in which the test was added + const char* m_description; + bool m_skip; + bool m_may_fail; + bool m_should_fail; + int m_expected_failures; + double m_timeout; + + DOCTEST_DECLARE_DEFAULTS(TestCaseData); + DOCTEST_DECLARE_COPIES(TestCaseData); +}; + +struct DOCTEST_INTERFACE AssertData +{ + // common - for all asserts + const TestCaseData* m_test_case; + assertType::Enum m_at; + const char* m_file; + int m_line; + const char* m_expr; + bool m_failed; + + // exception-related - for all asserts + bool m_threw; + String m_exception; + + // for normal asserts + String m_decomp; + + // for specific exception-related asserts + bool m_threw_as; + const char* m_exception_type; + + DOCTEST_DECLARE_DEFAULTS(AssertData); + DOCTEST_DELETE_COPIES(AssertData); +}; + +struct DOCTEST_INTERFACE MessageData +{ + String m_string; + const char* m_file; + int m_line; + assertType::Enum m_severity; + + DOCTEST_DECLARE_DEFAULTS(MessageData); + DOCTEST_DELETE_COPIES(MessageData); +}; + +struct DOCTEST_INTERFACE SubcaseSignature +{ + const char* m_name; + const char* m_file; + int m_line; + + SubcaseSignature(const char* name, const char* file, int line); + + bool operator<(const SubcaseSignature& other) const; + + DOCTEST_DECLARE_DEFAULTS(SubcaseSignature); + DOCTEST_DECLARE_COPIES(SubcaseSignature); +}; + +struct DOCTEST_INTERFACE IContextScope +{ + DOCTEST_DELETE_COPIES(IContextScope); + + IContextScope(); + virtual ~IContextScope(); + virtual void stringify(std::ostream*) const = 0; +}; + +struct ContextOptions //!OCLINT too many fields +{ + // == parameters from the command line + String order_by; // how tests should be ordered + unsigned rand_seed; // the seed for rand ordering + + unsigned first; // the first (matching) test to be executed + unsigned last; // the last (matching) test to be executed + + int abort_after; // stop tests after this many failed assertions + int subcase_filter_levels; // apply the subcase filters for the first N levels + + bool success; // include successful assertions in output + bool case_sensitive; // if filtering should be case sensitive + bool exit; // if the program should be exited after the tests are ran/whatever + bool duration; // print the time duration of each test case + bool no_throw; // to skip exceptions-related assertion macros + bool no_exitcode; // if the framework should return 0 as the exitcode + bool no_run; // to not run the tests at all (can be done with an "*" exclude) + bool no_version; // to not print the version of the framework + bool no_colors; // if output to the console should be colorized + bool force_colors; // forces the use of colors even when a tty cannot be detected + bool no_breaks; // to not break into the debugger + bool no_skip; // don't skip test cases which are marked to be skipped + bool gnu_file_line; // if line numbers should be surrounded with :x: and not (x): + bool no_path_in_filenames; // if the path to files should be removed from the output + bool no_line_numbers; // if source code line numbers should be omitted from the output + bool no_skipped_summary; // don't print "skipped" in the summary !!! UNDOCUMENTED !!! + + bool help; // to print the help + bool version; // to print the version + bool count; // if only the count of matching tests is to be retreived + bool list_test_cases; // to list all tests matching the filters + bool list_test_suites; // to list all suites matching the filters + bool list_reporters; // lists all registered reporters + + DOCTEST_DECLARE_DEFAULTS(ContextOptions); + DOCTEST_DELETE_COPIES(ContextOptions); +}; + +namespace detail { +#if defined(DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING) || defined(DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS) + template + struct enable_if + {}; + + template + struct enable_if + { typedef TYPE type; }; +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING) || DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + + // clang-format off + template struct remove_reference { typedef T type; }; + template struct remove_reference { typedef T type; }; + template struct remove_reference { typedef T type; }; + + template struct remove_const { typedef T type; }; + template struct remove_const { typedef T type; }; + // clang-format on + + template + struct deferred_false + // cppcheck-suppress unusedStructMember + { static const bool value = false; }; + + namespace has_insertion_operator_impl { + typedef char no; + typedef char yes[2]; + + struct any_t + { + template + // cppcheck-suppress noExplicitConstructor + any_t(const DOCTEST_REF_WRAP(T)); + }; + + yes& testStreamable(std::ostream&); + no testStreamable(no); + + no operator<<(const std::ostream&, const any_t&); + + template + struct has_insertion_operator + { + static std::ostream& s; + static const DOCTEST_REF_WRAP(T) t; + static const bool value = sizeof(testStreamable(s << t)) == sizeof(yes); + }; + } // namespace has_insertion_operator_impl + + template + struct has_insertion_operator : has_insertion_operator_impl::has_insertion_operator + {}; + + DOCTEST_INTERFACE void my_memcpy(void* dest, const void* src, unsigned num); + + DOCTEST_INTERFACE std::ostream* getTlsOss(); // returns a thread-local ostringstream + DOCTEST_INTERFACE String getTlsOssResult(); + + template + struct StringMakerBase + { + template + static String convert(const DOCTEST_REF_WRAP(T)) { + return "{?}"; + } + }; + + template <> + struct StringMakerBase + { + template + static String convert(const DOCTEST_REF_WRAP(T) in) { + *getTlsOss() << in; + return getTlsOssResult(); + } + }; + + DOCTEST_INTERFACE String rawMemoryToString(const void* object, unsigned size); + + template + String rawMemoryToString(const DOCTEST_REF_WRAP(T) object) { + return rawMemoryToString(&object, sizeof(object)); + } + + template + const char* type_to_string() { + return "<>"; + } +} // namespace detail + +template +struct StringMaker : public detail::StringMakerBase::value> +{}; + +template +struct StringMaker +{ + template + static String convert(U* p) { + if(p) + return detail::rawMemoryToString(p); + return "NULL"; + } +}; + +template +struct StringMaker +{ + static String convert(R C::*p) { + if(p) + return detail::rawMemoryToString(p); + return "NULL"; + } +}; + +template +String toString(const DOCTEST_REF_WRAP(T) value) { + return StringMaker::convert(value); +} + +#ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +DOCTEST_INTERFACE String toString(char* in); +DOCTEST_INTERFACE String toString(const char* in); +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +DOCTEST_INTERFACE String toString(bool in); +DOCTEST_INTERFACE String toString(float in); +DOCTEST_INTERFACE String toString(double in); +DOCTEST_INTERFACE String toString(double long in); + +DOCTEST_INTERFACE String toString(char in); +DOCTEST_INTERFACE String toString(char signed in); +DOCTEST_INTERFACE String toString(char unsigned in); +DOCTEST_INTERFACE String toString(int short in); +DOCTEST_INTERFACE String toString(int short unsigned in); +DOCTEST_INTERFACE String toString(int in); +DOCTEST_INTERFACE String toString(int unsigned in); +DOCTEST_INTERFACE String toString(int long in); +DOCTEST_INTERFACE String toString(int long unsigned in); +DOCTEST_INTERFACE String toString(int long long in); +DOCTEST_INTERFACE String toString(int long long unsigned in); +DOCTEST_INTERFACE String toString(std::nullptr_t in); + +class DOCTEST_INTERFACE Approx +{ +public: + explicit Approx(double value); + + DOCTEST_DECLARE_COPIES(Approx); + + Approx operator()(double value) const; + +#ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + template + explicit Approx(const T& value, + typename detail::enable_if::value>::type* = + static_cast(nullptr)) { + *this = Approx(static_cast(value)); + } +#endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + + Approx& epsilon(double newEpsilon); + +#ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + template + typename detail::enable_if::value, Approx&>::type epsilon( + const T& newEpsilon) { + m_epsilon = static_cast(newEpsilon); + return *this; + } +#endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + + Approx& scale(double newScale); + +#ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + template + typename detail::enable_if::value, Approx&>::type scale( + const T& newScale) { + m_scale = static_cast(newScale); + return *this; + } +#endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + + // clang-format off + DOCTEST_INTERFACE friend bool operator==(double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator==(const Approx & lhs, double rhs); + DOCTEST_INTERFACE friend bool operator!=(double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator!=(const Approx & lhs, double rhs); + DOCTEST_INTERFACE friend bool operator<=(double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator<=(const Approx & lhs, double rhs); + DOCTEST_INTERFACE friend bool operator>=(double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator>=(const Approx & lhs, double rhs); + DOCTEST_INTERFACE friend bool operator< (double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator< (const Approx & lhs, double rhs); + DOCTEST_INTERFACE friend bool operator> (double lhs, const Approx & rhs); + DOCTEST_INTERFACE friend bool operator> (const Approx & lhs, double rhs); + + DOCTEST_INTERFACE friend String toString(const Approx& in); + +#ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS +#define DOCTEST_APPROX_PREFIX \ + template friend typename detail::enable_if::value, bool>::type + + DOCTEST_APPROX_PREFIX operator==(const T& lhs, const Approx& rhs) { return operator==(double(lhs), rhs); } + DOCTEST_APPROX_PREFIX operator==(const Approx& lhs, const T& rhs) { return operator==(rhs, lhs); } + DOCTEST_APPROX_PREFIX operator!=(const T& lhs, const Approx& rhs) { return !operator==(lhs, rhs); } + DOCTEST_APPROX_PREFIX operator!=(const Approx& lhs, const T& rhs) { return !operator==(rhs, lhs); } + DOCTEST_APPROX_PREFIX operator<=(const T& lhs, const Approx& rhs) { return double(lhs) < rhs.m_value || lhs == rhs; } + DOCTEST_APPROX_PREFIX operator<=(const Approx& lhs, const T& rhs) { return lhs.m_value < double(rhs) || lhs == rhs; } + DOCTEST_APPROX_PREFIX operator>=(const T& lhs, const Approx& rhs) { return double(lhs) > rhs.m_value || lhs == rhs; } + DOCTEST_APPROX_PREFIX operator>=(const Approx& lhs, const T& rhs) { return lhs.m_value > double(rhs) || lhs == rhs; } + DOCTEST_APPROX_PREFIX operator< (const T& lhs, const Approx& rhs) { return double(lhs) < rhs.m_value && lhs != rhs; } + DOCTEST_APPROX_PREFIX operator< (const Approx& lhs, const T& rhs) { return lhs.m_value < double(rhs) && lhs != rhs; } + DOCTEST_APPROX_PREFIX operator> (const T& lhs, const Approx& rhs) { return double(lhs) > rhs.m_value && lhs != rhs; } + DOCTEST_APPROX_PREFIX operator> (const Approx& lhs, const T& rhs) { return lhs.m_value > double(rhs) && lhs != rhs; } +#undef DOCTEST_APPROX_PREFIX +#endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS + + // clang-format on + +private: + double m_epsilon; + double m_scale; + double m_value; +}; + +DOCTEST_INTERFACE String toString(const Approx& in); + +DOCTEST_INTERFACE const ContextOptions* getContextOptions(); + +#if !defined(DOCTEST_CONFIG_DISABLE) + +namespace detail { + // clang-format off +#ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + template struct decay_array { typedef T type; }; + template struct decay_array { typedef T* type; }; + template struct decay_array { typedef T* type; }; + + template struct not_char_pointer { enum { value = 1 }; }; + template<> struct not_char_pointer { enum { value = 0 }; }; + template<> struct not_char_pointer { enum { value = 0 }; }; + + template struct can_use_op : public not_char_pointer::type> {}; +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + // clang-format on + + struct DOCTEST_INTERFACE TestFailureException + { + DOCTEST_DECLARE_DEFAULTS(TestFailureException); + DOCTEST_DECLARE_COPIES(TestFailureException); + }; + + DOCTEST_INTERFACE bool checkIfShouldThrow(assertType::Enum at); + DOCTEST_INTERFACE void throwException(); + + struct DOCTEST_INTERFACE Subcase + { + SubcaseSignature m_signature; + bool m_entered = false; + + Subcase(const char* name, const char* file, int line); + ~Subcase(); + + DOCTEST_DELETE_COPIES(Subcase); + + operator bool() const; + }; + + template + String stringifyBinaryExpr(const DOCTEST_REF_WRAP(L) lhs, const char* op, + const DOCTEST_REF_WRAP(R) rhs) { + return toString(lhs) + op + toString(rhs); + } + +#define DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(op, op_str, op_macro) \ + template \ + DOCTEST_NOINLINE Result operator op(const DOCTEST_REF_WRAP(R) rhs) { \ + bool res = op_macro(lhs, rhs); \ + if(m_at & assertType::is_false) \ + res = !res; \ + if(!res || doctest::getContextOptions()->success) \ + return Result(res, stringifyBinaryExpr(lhs, op_str, rhs)); \ + return Result(res); \ + } + +#define DOCTEST_FORBIT_EXPRESSION(rt, op) \ + template \ + rt& operator op(const R&) { \ + static_assert(deferred_false::value, \ + "Expression Too Complex Please Rewrite As Binary Comparison!"); \ + return *this; \ + } + + struct DOCTEST_INTERFACE Result + { + bool m_passed; + String m_decomp; + + Result(bool passed, const String& decomposition = String()); + + DOCTEST_DECLARE_DEFAULTS(Result); + DOCTEST_DECLARE_COPIES(Result); + + // forbidding some expressions based on this table: http://en.cppreference.com/w/cpp/language/operator_precedence + DOCTEST_FORBIT_EXPRESSION(Result, &) + DOCTEST_FORBIT_EXPRESSION(Result, ^) + DOCTEST_FORBIT_EXPRESSION(Result, |) + DOCTEST_FORBIT_EXPRESSION(Result, &&) + DOCTEST_FORBIT_EXPRESSION(Result, ||) + DOCTEST_FORBIT_EXPRESSION(Result, ==) + DOCTEST_FORBIT_EXPRESSION(Result, !=) + DOCTEST_FORBIT_EXPRESSION(Result, <) + DOCTEST_FORBIT_EXPRESSION(Result, >) + DOCTEST_FORBIT_EXPRESSION(Result, <=) + DOCTEST_FORBIT_EXPRESSION(Result, >=) + DOCTEST_FORBIT_EXPRESSION(Result, =) + DOCTEST_FORBIT_EXPRESSION(Result, +=) + DOCTEST_FORBIT_EXPRESSION(Result, -=) + DOCTEST_FORBIT_EXPRESSION(Result, *=) + DOCTEST_FORBIT_EXPRESSION(Result, /=) + DOCTEST_FORBIT_EXPRESSION(Result, %=) + DOCTEST_FORBIT_EXPRESSION(Result, <<=) + DOCTEST_FORBIT_EXPRESSION(Result, >>=) + DOCTEST_FORBIT_EXPRESSION(Result, &=) + DOCTEST_FORBIT_EXPRESSION(Result, ^=) + DOCTEST_FORBIT_EXPRESSION(Result, |=) + }; + +#ifndef DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION + + DOCTEST_CLANG_SUPPRESS_WARNING_PUSH + DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-conversion") + DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-compare") + //DOCTEST_CLANG_SUPPRESS_WARNING("-Wdouble-promotion") + //DOCTEST_CLANG_SUPPRESS_WARNING("-Wconversion") + //DOCTEST_CLANG_SUPPRESS_WARNING("-Wfloat-equal") + + DOCTEST_GCC_SUPPRESS_WARNING_PUSH + DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-conversion") + DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-compare") + //DOCTEST_GCC_SUPPRESS_WARNING("-Wdouble-promotion") + //DOCTEST_GCC_SUPPRESS_WARNING("-Wconversion") + //DOCTEST_GCC_SUPPRESS_WARNING("-Wfloat-equal") + + DOCTEST_MSVC_SUPPRESS_WARNING_PUSH + // http://stackoverflow.com/questions/39479163 what's the difference between 4018 and 4389 + DOCTEST_MSVC_SUPPRESS_WARNING(4388) // signed/unsigned mismatch + DOCTEST_MSVC_SUPPRESS_WARNING(4389) // 'operator' : signed/unsigned mismatch + DOCTEST_MSVC_SUPPRESS_WARNING(4018) // 'expression' : signed/unsigned mismatch + //DOCTEST_MSVC_SUPPRESS_WARNING(4805) // 'operation' : unsafe mix of type 'type' and type 'type' in operation + +#endif // DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION + + // clang-format off +#ifndef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +#define DOCTEST_COMPARISON_RETURN_TYPE bool +#else // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +#define DOCTEST_COMPARISON_RETURN_TYPE typename enable_if::value || can_use_op::value, bool>::type + inline bool eq(const char* lhs, const char* rhs) { return String(lhs) == String(rhs); } + inline bool ne(const char* lhs, const char* rhs) { return String(lhs) != String(rhs); } + inline bool lt(const char* lhs, const char* rhs) { return String(lhs) < String(rhs); } + inline bool gt(const char* lhs, const char* rhs) { return String(lhs) > String(rhs); } + inline bool le(const char* lhs, const char* rhs) { return String(lhs) <= String(rhs); } + inline bool ge(const char* lhs, const char* rhs) { return String(lhs) >= String(rhs); } +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + // clang-format on + +#define DOCTEST_RELATIONAL_OP(name, op) \ + template \ + DOCTEST_COMPARISON_RETURN_TYPE name(const DOCTEST_REF_WRAP(L) lhs, \ + const DOCTEST_REF_WRAP(R) rhs) { \ + return lhs op rhs; \ + } + + DOCTEST_RELATIONAL_OP(eq, ==) + DOCTEST_RELATIONAL_OP(ne, !=) + DOCTEST_RELATIONAL_OP(lt, <) + DOCTEST_RELATIONAL_OP(gt, >) + DOCTEST_RELATIONAL_OP(le, <=) + DOCTEST_RELATIONAL_OP(ge, >=) + +#ifndef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +#define DOCTEST_CMP_EQ(l, r) l == r +#define DOCTEST_CMP_NE(l, r) l != r +#define DOCTEST_CMP_GT(l, r) l > r +#define DOCTEST_CMP_LT(l, r) l < r +#define DOCTEST_CMP_GE(l, r) l >= r +#define DOCTEST_CMP_LE(l, r) l <= r +#else // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +#define DOCTEST_CMP_EQ(l, r) eq(l, r) +#define DOCTEST_CMP_NE(l, r) ne(l, r) +#define DOCTEST_CMP_GT(l, r) gt(l, r) +#define DOCTEST_CMP_LT(l, r) lt(l, r) +#define DOCTEST_CMP_GE(l, r) ge(l, r) +#define DOCTEST_CMP_LE(l, r) le(l, r) +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + + template + // cppcheck-suppress copyCtorAndEqOperator + struct Expression_lhs + { + L lhs; + assertType::Enum m_at; + + explicit Expression_lhs(L in, assertType::Enum at) + : lhs(in) + , m_at(at) {} + + DOCTEST_NOINLINE operator Result() { + bool res = !!lhs; + if(m_at & assertType::is_false) //!OCLINT bitwise operator in conditional + res = !res; + + if(!res || getContextOptions()->success) + return Result(res, toString(lhs)); + return Result(res); + } + + // clang-format off + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(==, " == ", DOCTEST_CMP_EQ) //!OCLINT bitwise operator in conditional + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(!=, " != ", DOCTEST_CMP_NE) //!OCLINT bitwise operator in conditional + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(>, " > ", DOCTEST_CMP_GT) //!OCLINT bitwise operator in conditional + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(<, " < ", DOCTEST_CMP_LT) //!OCLINT bitwise operator in conditional + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(>=, " >= ", DOCTEST_CMP_GE) //!OCLINT bitwise operator in conditional + DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(<=, " <= ", DOCTEST_CMP_LE) //!OCLINT bitwise operator in conditional + // clang-format on + + // forbidding some expressions based on this table: http://en.cppreference.com/w/cpp/language/operator_precedence + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ^) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, |) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &&) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ||) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, =) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, +=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, -=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, *=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, /=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, %=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, <<=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, >>=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ^=) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, |=) + // these 2 are unfortunate because they should be allowed - they have higher precedence over the comparisons, but the + // ExpressionDecomposer class uses the left shift operator to capture the left operand of the binary expression... + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, <<) + DOCTEST_FORBIT_EXPRESSION(Expression_lhs, >>) + }; + +#ifndef DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION + + DOCTEST_CLANG_SUPPRESS_WARNING_POP + DOCTEST_MSVC_SUPPRESS_WARNING_POP + DOCTEST_GCC_SUPPRESS_WARNING_POP + +#endif // DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION + + struct DOCTEST_INTERFACE ExpressionDecomposer + { + assertType::Enum m_at; + + ExpressionDecomposer(assertType::Enum at); + + DOCTEST_DECLARE_DEFAULTS(ExpressionDecomposer); + DOCTEST_DELETE_COPIES(ExpressionDecomposer); + + // The right operator for capturing expressions is "<=" instead of "<<" (based on the operator precedence table) + // but then there will be warnings from GCC about "-Wparentheses" and since "_Pragma()" is problematic this will stay for now... + // https://github.com/philsquared/Catch/issues/870 + // https://github.com/philsquared/Catch/issues/565 + template + Expression_lhs operator<<(const DOCTEST_REF_WRAP(L) operand) { + return Expression_lhs(operand, m_at); + } + }; + + struct DOCTEST_INTERFACE TestSuite + { + const char* m_test_suite; + const char* m_description; + bool m_skip; + bool m_may_fail; + bool m_should_fail; + int m_expected_failures; + double m_timeout; + + DOCTEST_DECLARE_DEFAULTS(TestSuite); + DOCTEST_DECLARE_COPIES(TestSuite); + + TestSuite& operator*(const char* in); + + template + TestSuite& operator*(const T& in) { + in.fill(*this); + return *this; + } + }; + + typedef void (*funcType)(); + + struct DOCTEST_INTERFACE TestCase : public TestCaseData + { + funcType m_test; // a function pointer to the test case + + const char* m_type; // for templated test cases - gets appended to the real name + int m_template_id; // an ID used to distinguish between the different versions of a templated test case + String m_full_name; // contains the name (only for templated test cases!) + the template type + + TestCase(funcType test, const char* file, unsigned line, const TestSuite& test_suite, + const char* type = "", int template_id = -1); + + DOCTEST_DECLARE_DEFAULTS(TestCase); + + TestCase(const TestCase& other); + + DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(26434) // hides a non-virtual function + TestCase& operator=(const TestCase& other); + DOCTEST_MSVC_SUPPRESS_WARNING_POP + + TestCase& operator*(const char* in); + + template + TestCase& operator*(const T& in) { + in.fill(*this); + return *this; + } + + bool operator<(const TestCase& other) const; + }; + + // forward declarations of functions used by the macros + DOCTEST_INTERFACE int regTest(const TestCase& tc); + DOCTEST_INTERFACE int setTestSuite(const TestSuite& ts); + DOCTEST_INTERFACE bool isDebuggerActive(); + + namespace binaryAssertComparison { + enum Enum + { + eq = 0, + ne, + gt, + lt, + ge, + le + }; + } // namespace binaryAssertComparison + + // clang-format off + template struct RelationalComparator { bool operator()(const DOCTEST_REF_WRAP(L), const DOCTEST_REF_WRAP(R) ) const { return false; } }; + +#define DOCTEST_BINARY_RELATIONAL_OP(n, op) \ + template struct RelationalComparator { bool operator()(const DOCTEST_REF_WRAP(L) lhs, const DOCTEST_REF_WRAP(R) rhs) const { return op(lhs, rhs); } }; + // clang-format on + + DOCTEST_BINARY_RELATIONAL_OP(0, eq) + DOCTEST_BINARY_RELATIONAL_OP(1, ne) + DOCTEST_BINARY_RELATIONAL_OP(2, gt) + DOCTEST_BINARY_RELATIONAL_OP(3, lt) + DOCTEST_BINARY_RELATIONAL_OP(4, ge) + DOCTEST_BINARY_RELATIONAL_OP(5, le) + + struct DOCTEST_INTERFACE ResultBuilder : public AssertData + { + ResultBuilder(assertType::Enum at, const char* file, int line, const char* expr, + const char* exception_type = ""); + + DOCTEST_DECLARE_DEFAULTS(ResultBuilder); + DOCTEST_DELETE_COPIES(ResultBuilder); + + void setResult(const Result& res); + + template + DOCTEST_NOINLINE void binary_assert(const DOCTEST_REF_WRAP(L) lhs, + const DOCTEST_REF_WRAP(R) rhs) { + m_failed = !RelationalComparator()(lhs, rhs); + if(m_failed || getContextOptions()->success) + m_decomp = stringifyBinaryExpr(lhs, ", ", rhs); + } + + template + DOCTEST_NOINLINE void unary_assert(const DOCTEST_REF_WRAP(L) val) { + m_failed = !val; + + if(m_at & assertType::is_false) //!OCLINT bitwise operator in conditional + m_failed = !m_failed; + + if(m_failed || getContextOptions()->success) + m_decomp = toString(val); + } + + void translateException(); + + bool log(); + void react() const; + }; + + namespace assertAction { + enum Enum + { + nothing = 0, + dbgbreak = 1, + shouldthrow = 2 + }; + } // namespace assertAction + + DOCTEST_INTERFACE void failed_out_of_a_testing_context(const AssertData& ad); + + DOCTEST_INTERFACE void decomp_assert(assertType::Enum at, const char* file, int line, + const char* expr, Result result); + +#define DOCTEST_ASSERT_OUT_OF_TESTS(decomp) \ + do { \ + if(!is_running_in_test) { \ + if(failed) { \ + ResultBuilder rb(at, file, line, expr); \ + rb.m_failed = failed; \ + rb.m_decomp = decomp; \ + failed_out_of_a_testing_context(rb); \ + if(isDebuggerActive() && !getContextOptions()->no_breaks) \ + DOCTEST_BREAK_INTO_DEBUGGER(); \ + if(checkIfShouldThrow(at)) \ + throwException(); \ + } \ + return; \ + } \ + } while(false) + +#define DOCTEST_ASSERT_IN_TESTS(decomp) \ + ResultBuilder rb(at, file, line, expr); \ + rb.m_failed = failed; \ + if(rb.m_failed || getContextOptions()->success) \ + rb.m_decomp = decomp; \ + if(rb.log()) \ + DOCTEST_BREAK_INTO_DEBUGGER(); \ + if(rb.m_failed && checkIfShouldThrow(at)) \ + throwException() + + template + DOCTEST_NOINLINE void binary_assert(assertType::Enum at, const char* file, int line, + const char* expr, const DOCTEST_REF_WRAP(L) lhs, + const DOCTEST_REF_WRAP(R) rhs) { + bool failed = !RelationalComparator()(lhs, rhs); + + // ################################################################################### + // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT + // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED + // ################################################################################### + DOCTEST_ASSERT_OUT_OF_TESTS(stringifyBinaryExpr(lhs, ", ", rhs)); + DOCTEST_ASSERT_IN_TESTS(stringifyBinaryExpr(lhs, ", ", rhs)); + } + + template + DOCTEST_NOINLINE void unary_assert(assertType::Enum at, const char* file, int line, + const char* expr, const DOCTEST_REF_WRAP(L) val) { + bool failed = !val; + + if(at & assertType::is_false) //!OCLINT bitwise operator in conditional + failed = !failed; + + // ################################################################################### + // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT + // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED + // ################################################################################### + DOCTEST_ASSERT_OUT_OF_TESTS(toString(val)); + DOCTEST_ASSERT_IN_TESTS(toString(val)); + } + + struct DOCTEST_INTERFACE IExceptionTranslator + { + DOCTEST_DELETE_COPIES(IExceptionTranslator); + + IExceptionTranslator(); + virtual ~IExceptionTranslator(); + virtual bool translate(String&) const = 0; + }; + + template + class ExceptionTranslator : public IExceptionTranslator //!OCLINT destructor of virtual class + { + public: + explicit ExceptionTranslator(String (*translateFunction)(T)) + : m_translateFunction(translateFunction) {} + + bool translate(String& res) const { +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS + try { + throw; + // cppcheck-suppress catchExceptionByValue + } catch(T ex) { // NOLINT + res = m_translateFunction(ex); //!OCLINT parameter reassignment + return true; + } catch(...) {} //!OCLINT - empty catch statement +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + ((void)res); // to silence -Wunused-parameter + return false; + } + + private: + String (*m_translateFunction)(T); + }; + + DOCTEST_INTERFACE void registerExceptionTranslatorImpl(const IExceptionTranslator* et); + + // FIX FOR VISUAL STUDIO VERSIONS PRIOR TO 2015 - they failed to compile the call to operator<< with + // std::ostream passed as a reference noting that there is a use of an undefined type (which there isn't) + DOCTEST_INTERFACE void writeStringToStream(std::ostream* s, const String& str); + + template + struct StringStreamBase + { + template + static void convert(std::ostream* s, const T& in) { + writeStringToStream(s, toString(in)); + } + + // always treat char* as a string in this context - no matter + // if DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING is defined + static void convert(std::ostream* s, const char* in) { writeStringToStream(s, String(in)); } + }; + + template <> + struct StringStreamBase + { + template + static void convert(std::ostream* s, const T& in) { + *s << in; + } + }; + + template + struct StringStream : public StringStreamBase::value> + {}; + + template + void toStream(std::ostream* s, const T& value) { + StringStream::convert(s, value); + } + +#ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + DOCTEST_INTERFACE void toStream(std::ostream* s, char* in); + DOCTEST_INTERFACE void toStream(std::ostream* s, const char* in); +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + DOCTEST_INTERFACE void toStream(std::ostream* s, bool in); + DOCTEST_INTERFACE void toStream(std::ostream* s, float in); + DOCTEST_INTERFACE void toStream(std::ostream* s, double in); + DOCTEST_INTERFACE void toStream(std::ostream* s, double long in); + + DOCTEST_INTERFACE void toStream(std::ostream* s, char in); + DOCTEST_INTERFACE void toStream(std::ostream* s, char signed in); + DOCTEST_INTERFACE void toStream(std::ostream* s, char unsigned in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int short in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int short unsigned in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int unsigned in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int long in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int long unsigned in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int long long in); + DOCTEST_INTERFACE void toStream(std::ostream* s, int long long unsigned in); + + class DOCTEST_INTERFACE ContextBuilder + { + friend class ContextScope; + + struct DOCTEST_INTERFACE ICapture + { + DOCTEST_DELETE_COPIES(ICapture); + ICapture(); + virtual ~ICapture(); + virtual void toStream(std::ostream*) const = 0; + }; + + template + struct Capture : public ICapture //!OCLINT destructor of virtual class + { + const T* capture; + + explicit Capture(const T* in) + : capture(in) {} + void toStream(std::ostream* s) const override { detail::toStream(s, *capture); } + }; + + struct DOCTEST_INTERFACE Chunk + { + char buf[sizeof(Capture)] DOCTEST_ALIGNMENT( + 2 * sizeof(void*)); // place to construct a Capture + + DOCTEST_DECLARE_DEFAULTS(Chunk); + DOCTEST_DELETE_COPIES(Chunk); + }; + + struct DOCTEST_INTERFACE Node + { + Chunk chunk; + Node* next; + + DOCTEST_DECLARE_DEFAULTS(Node); + DOCTEST_DELETE_COPIES(Node); + }; + + Chunk stackChunks[DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK]; + int numCaptures = 0; + Node* head = nullptr; + Node* tail = nullptr; + + ContextBuilder(ContextBuilder& other); + + ContextBuilder& operator=(const ContextBuilder&) = delete; + + void stringify(std::ostream* s) const; + + public: + ContextBuilder(); + ~ContextBuilder(); + + template + DOCTEST_NOINLINE ContextBuilder& operator<<(T& in) { + Capture temp(&in); + + // construct either on stack or on heap + // copy the bytes for the whole object - including the vtable because we cant construct + // the object directly in the buffer using placement new - need the header... + if(numCaptures < DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK) { + my_memcpy(stackChunks[numCaptures].buf, &temp, sizeof(Chunk)); + } else { + auto curr = new Node; + curr->next = nullptr; + if(tail) { + tail->next = curr; + tail = curr; + } else { + head = tail = curr; + } + + my_memcpy(tail->chunk.buf, &temp, sizeof(Chunk)); + } + ++numCaptures; + return *this; + } + + template + ContextBuilder& operator<<(const T&&) { + static_assert(deferred_false::value, + "Cannot pass temporaries or rvalues to the streaming operator because it " + "caches pointers to the passed objects for lazy evaluation!"); + return *this; + } + }; + + class DOCTEST_INTERFACE ContextScope : public IContextScope + { + ContextBuilder contextBuilder; + + public: + explicit ContextScope(ContextBuilder& temp); + + DOCTEST_DELETE_COPIES(ContextScope); + + ~ContextScope(); + + void stringify(std::ostream* s) const; + }; + + struct DOCTEST_INTERFACE MessageBuilder : public MessageData + { + std::ostream* m_stream; + + MessageBuilder(const char* file, int line, assertType::Enum severity); + MessageBuilder() = delete; + ~MessageBuilder(); + + DOCTEST_DELETE_COPIES(MessageBuilder); + + template + MessageBuilder& operator<<(const T& in) { + toStream(m_stream, in); + return *this; + } + + bool log(); + void react(); + }; +} // namespace detail + +#define DOCTEST_DEFINE_DECORATOR(name, type, def) \ + struct name \ + { \ + type data; \ + name(type in = def) \ + : data(in) {} \ + void fill(detail::TestCase& state) const { state.DOCTEST_CAT(m_, name) = data; } \ + void fill(detail::TestSuite& state) const { state.DOCTEST_CAT(m_, name) = data; } \ + } + +DOCTEST_DEFINE_DECORATOR(test_suite, const char*, ""); +DOCTEST_DEFINE_DECORATOR(description, const char*, ""); +DOCTEST_DEFINE_DECORATOR(skip, bool, true); +DOCTEST_DEFINE_DECORATOR(timeout, double, 0); +DOCTEST_DEFINE_DECORATOR(may_fail, bool, true); +DOCTEST_DEFINE_DECORATOR(should_fail, bool, true); +DOCTEST_DEFINE_DECORATOR(expected_failures, int, 0); + +template +int registerExceptionTranslator(String (*translateFunction)(T)) { + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wexit-time-destructors") + static detail::ExceptionTranslator exceptionTranslator(translateFunction); + DOCTEST_CLANG_SUPPRESS_WARNING_POP + detail::registerExceptionTranslatorImpl(&exceptionTranslator); + return 0; +} + +} // namespace doctest + +// in a separate namespace outside of doctest because the DOCTEST_TEST_SUITE macro +// introduces an anonymous namespace in which getCurrentTestSuite gets overridden +namespace doctest_detail_test_suite_ns { +DOCTEST_INTERFACE doctest::detail::TestSuite& getCurrentTestSuite(); +} // namespace doctest_detail_test_suite_ns + +namespace doctest { +#else // DOCTEST_CONFIG_DISABLE +template +int registerExceptionTranslator(String (*)(T)) { + return 0; +} +#endif // DOCTEST_CONFIG_DISABLE + +namespace detail { + typedef void (*assert_handler)(const AssertData&); + struct ContextState; +} // namespace detail + +class DOCTEST_INTERFACE Context +{ + detail::ContextState* p; + + void parseArgs(int argc, const char* const* argv, bool withDefaults = false); + +public: + explicit Context(int argc = 0, const char* const* argv = nullptr); + + DOCTEST_DELETE_COPIES(Context); + + ~Context(); + + void applyCommandLine(int argc, const char* const* argv); + + void addFilter(const char* filter, const char* value); + void clearFilters(); + void setOption(const char* option, int value); + void setOption(const char* option, const char* value); + + bool shouldExit(); + + void setAsDefaultForAssertsOutOfTestCases(); + + void setAssertHandler(detail::assert_handler ah); + + int run(); +}; + +namespace TestCaseFailureReason { + enum Enum + { + None = 0, + AssertFailure = 1, // an assertion has failed in the test case + Exception = 2, // test case threw an exception + Crash = 4, // a crash... + TooManyFailedAsserts = 8, // the abort-after option + Timeout = 16, // see the timeout decorator + ShouldHaveFailedButDidnt = 32, // see the should_fail decorator + ShouldHaveFailedAndDid = 64, // see the should_fail decorator + DidntFailExactlyNumTimes = 128, // see the expected_failures decorator + FailedExactlyNumTimes = 256, // see the expected_failures decorator + CouldHaveFailedAndDid = 512 // see the may_fail decorator + }; +} // namespace TestCaseFailureReason + +struct DOCTEST_INTERFACE CurrentTestCaseStats +{ + int numAssertsForCurrentTestCase; + int numAssertsFailedForCurrentTestCase; + double seconds_so_far; + int failure_flags; // use TestCaseFailureReason::Enum + String error_string; + bool should_reenter; // means we are not done with the test case because of subcases + + DOCTEST_DECLARE_DEFAULTS(CurrentTestCaseStats); + DOCTEST_DELETE_COPIES(CurrentTestCaseStats); +}; + +struct DOCTEST_INTERFACE TestRunStats +{ + unsigned numTestCases; + unsigned numTestCasesPassingFilters; + unsigned numTestSuitesPassingFilters; + unsigned numTestCasesFailed; + int numAsserts; + int numAssertsFailed; + + DOCTEST_DECLARE_DEFAULTS(TestRunStats); + DOCTEST_DELETE_COPIES(TestRunStats); +}; + +struct DOCTEST_INTERFACE IReporter +{ + // called when the whole test run starts (safe to cache a pointer to the input) + virtual void test_run_start(const ContextOptions&) = 0; + // called when the whole test run ends (caching a pointer to the input doesn't make sense here) + virtual void test_run_end(const TestRunStats&) = 0; + + // called when a test case is started (safe to cache a pointer to the input) + virtual void test_case_start(const TestCaseData&) = 0; + // called when a test case has ended - could be re-entered if more subcases have to be + // traversed - check CurrentTestCaseStats::should_reenter (caching a pointer to the input doesn't make sense here) + virtual void test_case_end(const CurrentTestCaseStats&) = 0; + + // called whenever a subcase is entered (don't cache pointers to the input) + virtual void subcase_start(const SubcaseSignature&) = 0; + // called whenever a subcase is exited (don't cache pointers to the input) + virtual void subcase_end(const SubcaseSignature&) = 0; + + // called for each assert (don't cache pointers to the input) + virtual void log_assert(const AssertData&) = 0; + // called for each message (don't cache pointers to the input) + virtual void log_message(const MessageData&) = 0; + + // called when a test case is skipped either because it doesn't pass the filters, has a skip decorator + // or isn't in the execution range (between first and last) (safe to cache a pointer to the input) + virtual void test_case_skipped(const TestCaseData&) = 0; + + // doctest will not be managing the lifetimes of reporters given to it but this would still be nice to have + virtual ~IReporter(); + + // can obtain all currently active contexts and stringify them if one wishes to do so + static int get_num_active_contexts(); + static const IContextScope* const* get_active_contexts(); + + // can iterate through contexts which have been stringified automatically in their destructors when an exception has been thrown + static int get_num_stringified_contexts(); + static const String* get_stringified_contexts(); +}; + +int registerReporter(const char* name, int priority, IReporter& r); + +} // namespace doctest + +// if registering is not disabled +#if !defined(DOCTEST_CONFIG_DISABLE) + +// common code in asserts - for convenience +#define DOCTEST_ASSERT_LOG_AND_REACT(b) \ + if(b.log()) \ + DOCTEST_BREAK_INTO_DEBUGGER(); \ + b.react() + +#ifdef DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS +#define DOCTEST_WRAP_IN_TRY(x) x; +#else // DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS +#define DOCTEST_WRAP_IN_TRY(x) \ + try { \ + x; \ + } catch(...) { _DOCTEST_RB.translateException(); } +#endif // DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS + +// registers the test by initializing a dummy var with a function +#define DOCTEST_REGISTER_FUNCTION(f, decorators) \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) = doctest::detail::regTest( \ + doctest::detail::TestCase(f, __FILE__, __LINE__, \ + doctest_detail_test_suite_ns::getCurrentTestSuite()) * \ + decorators); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() + +#define DOCTEST_IMPLEMENT_FIXTURE(der, base, func, decorators) \ + namespace { \ + struct der : public base \ + { \ + void f(); \ + }; \ + static void func() { \ + der v; \ + v.f(); \ + } \ + DOCTEST_REGISTER_FUNCTION(func, decorators) \ + } \ + inline DOCTEST_NOINLINE void der::f() + +#define DOCTEST_CREATE_AND_REGISTER_FUNCTION(f, decorators) \ + static void f(); \ + DOCTEST_REGISTER_FUNCTION(f, decorators) \ + static void f() + +// for registering tests +#define DOCTEST_TEST_CASE(decorators) \ + DOCTEST_CREATE_AND_REGISTER_FUNCTION(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), decorators) + +// for registering tests with a fixture +#define DOCTEST_TEST_CASE_FIXTURE(c, decorators) \ + DOCTEST_IMPLEMENT_FIXTURE(DOCTEST_ANONYMOUS(_DOCTEST_ANON_CLASS_), c, \ + DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), decorators) + +// for converting types to strings without the header and demangling +#define DOCTEST_TYPE_TO_STRING_IMPL(...) \ + template <> \ + inline const char* type_to_string<__VA_ARGS__>() { \ + return "<" #__VA_ARGS__ ">"; \ + } +#define DOCTEST_TYPE_TO_STRING(...) \ + namespace doctest { namespace detail { \ + DOCTEST_TYPE_TO_STRING_IMPL(__VA_ARGS__) \ + } \ + } \ + typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for typed tests +#define DOCTEST_REGISTER_TYPED_TEST_CASE_IMPL(func, type, decorators, idx) \ + doctest::detail::regTest( \ + doctest::detail::TestCase(func, __FILE__, __LINE__, \ + doctest_detail_test_suite_ns::getCurrentTestSuite(), \ + doctest::detail::type_to_string(), idx) * \ + decorators) + +#define DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL(dec, T, id, anon) \ + template \ + inline void anon(); \ + template \ + struct DOCTEST_CAT(id, ITERATOR) \ + { \ + DOCTEST_CAT(id, ITERATOR)(int line, int index) { \ + DOCTEST_REGISTER_TYPED_TEST_CASE_IMPL(anon, Type, dec, line * 1000 + index); \ + DOCTEST_CAT(id, ITERATOR)(line, index + 1); \ + } \ + }; \ + template \ + struct DOCTEST_CAT(id, ITERATOR) \ + { \ + DOCTEST_CAT(id, ITERATOR)(int line, int index) { \ + DOCTEST_REGISTER_TYPED_TEST_CASE_IMPL(anon, Type, dec, line * 1000 + index); \ + } \ + } + +#define DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL_PROXY(dec, T, id, anon) \ + DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL(dec, T, id, anon); \ + template \ + inline void anon() + +#define DOCTEST_TEST_CASE_TEMPLATE_DEFINE(dec, T, id) \ + DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL_PROXY(dec, T, id, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_)) + +#define DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(id, anon, ...) \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_CAT(anon, DUMMY)) = []() { \ + DOCTEST_CAT(id, ITERATOR)<__VA_ARGS__> DOCTEST_UNUSED DOCTEST_CAT(anon, inner_dummy)( \ + __LINE__, 0); \ + return 0; \ + }(); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() + +#define DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE(id, ...) \ + DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(id, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_), \ + __VA_ARGS__) \ + typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +#define DOCTEST_TEST_CASE_TEMPLATE_IMPL(dec, T, anon, ...) \ + DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL_PROXY(dec, T, anon, anon); \ + DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(anon, anon, __VA_ARGS__) \ + template \ + inline void anon() + +#define DOCTEST_TEST_CASE_TEMPLATE(dec, T, ...) \ + DOCTEST_TEST_CASE_TEMPLATE_IMPL(dec, T, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_), __VA_ARGS__) + +// for subcases +#define DOCTEST_SUBCASE(name) \ + if(const doctest::detail::Subcase & DOCTEST_ANONYMOUS(_DOCTEST_ANON_SUBCASE_) DOCTEST_UNUSED = \ + doctest::detail::Subcase(name, __FILE__, __LINE__)) + +// for grouping tests in test suites by using code blocks +#define DOCTEST_TEST_SUITE_IMPL(decorators, ns_name) \ + namespace ns_name { namespace doctest_detail_test_suite_ns { \ + static DOCTEST_NOINLINE doctest::detail::TestSuite& getCurrentTestSuite() { \ + DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4640) \ + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wexit-time-destructors") \ + static doctest::detail::TestSuite data; \ + static bool inited = false; \ + DOCTEST_MSVC_SUPPRESS_WARNING_POP \ + DOCTEST_CLANG_SUPPRESS_WARNING_POP \ + if(!inited) { \ + data* decorators; \ + inited = true; \ + } \ + return data; \ + } \ + } \ + } \ + namespace ns_name + +#define DOCTEST_TEST_SUITE(decorators) \ + DOCTEST_TEST_SUITE_IMPL(decorators, DOCTEST_ANONYMOUS(_DOCTEST_ANON_SUITE_)) + +// for starting a testsuite block +#define DOCTEST_TEST_SUITE_BEGIN(decorators) \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) = \ + doctest::detail::setTestSuite(doctest::detail::TestSuite() * decorators); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() \ + typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for ending a testsuite block +#define DOCTEST_TEST_SUITE_END \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) = \ + doctest::detail::setTestSuite(doctest::detail::TestSuite() * ""); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() \ + typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for registering exception translators +#define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR_IMPL(translatorName, signature) \ + inline doctest::String translatorName(signature); \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_)) = \ + doctest::registerExceptionTranslator(translatorName); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() \ + doctest::String translatorName(signature) + +#define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR(signature) \ + DOCTEST_REGISTER_EXCEPTION_TRANSLATOR_IMPL(DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_), \ + signature) + +// for registering +#define DOCTEST_REGISTER_REPORTER(name, priority, reporter) \ + DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_REPORTER_)) = \ + doctest::registerReporter(name, priority, reporter); \ + DOCTEST_GLOBAL_NO_WARNINGS_END() typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for logging +#define DOCTEST_INFO(x) \ + doctest::detail::ContextScope DOCTEST_ANONYMOUS(_DOCTEST_CAPTURE_)( \ + doctest::detail::ContextBuilder() << x) +#define DOCTEST_CAPTURE(x) DOCTEST_INFO(#x " := " << x) + +#define DOCTEST_ADD_AT_IMPL(type, file, line, mb, x) \ + do { \ + doctest::detail::MessageBuilder mb(file, line, doctest::assertType::type); \ + mb << x; \ + DOCTEST_ASSERT_LOG_AND_REACT(mb); \ + } while((void)0, 0) + +// clang-format off +#define DOCTEST_ADD_MESSAGE_AT(file, line, x) DOCTEST_ADD_AT_IMPL(is_warn, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), x) +#define DOCTEST_ADD_FAIL_CHECK_AT(file, line, x) DOCTEST_ADD_AT_IMPL(is_check, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), x) +#define DOCTEST_ADD_FAIL_AT(file, line, x) DOCTEST_ADD_AT_IMPL(is_require, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), x) +// clang-format on + +#define DOCTEST_MESSAGE(x) DOCTEST_ADD_MESSAGE_AT(__FILE__, __LINE__, x) +#define DOCTEST_FAIL_CHECK(x) DOCTEST_ADD_FAIL_CHECK_AT(__FILE__, __LINE__, x) +#define DOCTEST_FAIL(x) DOCTEST_ADD_FAIL_AT(__FILE__, __LINE__, x) + +// hack for macros like INFO() that require lvalues +#if __cplusplus >= 201402L || (DOCTEST_MSVC >= DOCTEST_COMPILER(19, 10, 0)) +template +constexpr T to_lvalue = x; +#define DOCTEST_TO_LVALUE(...) to_lvalue +#else // TO_LVALUE +#define DOCTEST_TO_LVALUE(...) TO_LVALUE_CAN_BE_USED_ONLY_IN_CPP14_MODE_OR_WITH_VS_2017_OR_NEWER +#endif // TO_LVALUE + +#ifndef DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_ASSERT_IMPLEMENT_2(assert_type, ...) \ + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Woverloaded-shift-op-parentheses") \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #__VA_ARGS__); \ + DOCTEST_WRAP_IN_TRY(_DOCTEST_RB.setResult( \ + doctest::detail::ExpressionDecomposer(doctest::assertType::assert_type) \ + << __VA_ARGS__)) \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB) \ + DOCTEST_CLANG_SUPPRESS_WARNING_POP + +#define DOCTEST_ASSERT_IMPLEMENT_1(assert_type, ...) \ + do { \ + DOCTEST_ASSERT_IMPLEMENT_2(assert_type, __VA_ARGS__); \ + } while((void)0, 0) + +#else // DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_ASSERT_IMPLEMENT_1(assert_type, ...) \ + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Woverloaded-shift-op-parentheses") \ + doctest::detail::decomp_assert( \ + doctest::assertType::assert_type, __FILE__, __LINE__, #__VA_ARGS__, \ + doctest::detail::ExpressionDecomposer(doctest::assertType::assert_type) \ + << __VA_ARGS__) DOCTEST_CLANG_SUPPRESS_WARNING_POP + +#endif // DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_WARN(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_WARN, __VA_ARGS__) +#define DOCTEST_CHECK(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_CHECK, __VA_ARGS__) +#define DOCTEST_REQUIRE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_REQUIRE, __VA_ARGS__) +#define DOCTEST_WARN_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_WARN_FALSE, __VA_ARGS__) +#define DOCTEST_CHECK_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_CHECK_FALSE, __VA_ARGS__) +#define DOCTEST_REQUIRE_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_REQUIRE_FALSE, __VA_ARGS__) + +// clang-format off +#define DOCTEST_WARN_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_WARN, cond); } while((void)0, 0) +#define DOCTEST_CHECK_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_CHECK, cond); } while((void)0, 0) +#define DOCTEST_REQUIRE_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_REQUIRE, cond); } while((void)0, 0) +#define DOCTEST_WARN_FALSE_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_WARN_FALSE, cond); } while((void)0, 0) +#define DOCTEST_CHECK_FALSE_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_CHECK_FALSE, cond); } while((void)0, 0) +#define DOCTEST_REQUIRE_FALSE_MESSAGE(cond, msg) do { DOCTEST_INFO(msg); DOCTEST_ASSERT_IMPLEMENT_2(DT_REQUIRE_FALSE, cond); } while((void)0, 0) +// clang-format on + +#define DOCTEST_ASSERT_THROWS(expr, assert_type) \ + do { \ + if(!doctest::getContextOptions()->no_throw) { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #expr); \ + try { \ + expr; \ + } catch(...) { _DOCTEST_RB.m_threw = true; } \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } \ + } while((void)0, 0) + +#define DOCTEST_ASSERT_THROWS_AS(expr, assert_type, ...) \ + do { \ + if(!doctest::getContextOptions()->no_throw) { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #expr, #__VA_ARGS__); \ + try { \ + expr; \ + } catch(const doctest::detail::remove_const< \ + doctest::detail::remove_reference<__VA_ARGS__>::type>::type&) { \ + _DOCTEST_RB.m_threw = true; \ + _DOCTEST_RB.m_threw_as = true; \ + } catch(...) { _DOCTEST_RB.translateException(); } \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } \ + } while((void)0, 0) + +#define DOCTEST_ASSERT_THROWS_WITH(expr, assert_type, ...) \ + do { \ + if(!doctest::getContextOptions()->no_throw) { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #expr, __VA_ARGS__); \ + try { \ + expr; \ + } catch(...) { _DOCTEST_RB.translateException(); } \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } \ + } while((void)0, 0) + +#define DOCTEST_ASSERT_NOTHROW(expr, assert_type) \ + do { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #expr); \ + try { \ + expr; \ + } catch(...) { _DOCTEST_RB.translateException(); } \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } while((void)0, 0) + +// clang-format off +#define DOCTEST_WARN_THROWS(expr) DOCTEST_ASSERT_THROWS(expr, DT_WARN_THROWS) +#define DOCTEST_CHECK_THROWS(expr) DOCTEST_ASSERT_THROWS(expr, DT_CHECK_THROWS) +#define DOCTEST_REQUIRE_THROWS(expr) DOCTEST_ASSERT_THROWS(expr, DT_REQUIRE_THROWS) + +#define DOCTEST_WARN_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_WARN_THROWS_AS, __VA_ARGS__) +#define DOCTEST_CHECK_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_CHECK_THROWS_AS, __VA_ARGS__) +#define DOCTEST_REQUIRE_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_REQUIRE_THROWS_AS, __VA_ARGS__) + +#define DOCTEST_WARN_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, DT_WARN_THROWS_WITH, __VA_ARGS__) +#define DOCTEST_CHECK_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, DT_CHECK_THROWS_WITH, __VA_ARGS__) +#define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, DT_REQUIRE_THROWS_WITH, __VA_ARGS__) + +#define DOCTEST_WARN_NOTHROW(expr) DOCTEST_ASSERT_NOTHROW(expr, DT_WARN_NOTHROW) +#define DOCTEST_CHECK_NOTHROW(expr) DOCTEST_ASSERT_NOTHROW(expr, DT_CHECK_NOTHROW) +#define DOCTEST_REQUIRE_NOTHROW(expr) DOCTEST_ASSERT_NOTHROW(expr, DT_REQUIRE_NOTHROW) + +#define DOCTEST_WARN_THROWS_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_WARN_THROWS(expr); } while((void)0, 0) +#define DOCTEST_CHECK_THROWS_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_CHECK_THROWS(expr); } while((void)0, 0) +#define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_REQUIRE_THROWS(expr); } while((void)0, 0) +#define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_WARN_THROWS_AS(expr, ex); } while((void)0, 0) +#define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_CHECK_THROWS_AS(expr, ex); } while((void)0, 0) +#define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_REQUIRE_THROWS_AS(expr, ex); } while((void)0, 0) +#define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_WARN_THROWS_WITH(expr, ex); } while((void)0, 0) +#define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_CHECK_THROWS_WITH(expr, ex); } while((void)0, 0) +#define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, ex, msg) do { DOCTEST_INFO(msg); DOCTEST_REQUIRE_THROWS_WITH(expr, ex); } while((void)0, 0) +#define DOCTEST_WARN_NOTHROW_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_WARN_NOTHROW(expr); } while((void)0, 0) +#define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_CHECK_NOTHROW(expr); } while((void)0, 0) +#define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, msg) do { DOCTEST_INFO(msg); DOCTEST_REQUIRE_NOTHROW(expr); } while((void)0, 0) +// clang-format on + +#ifndef DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_BINARY_ASSERT(assert_type, comp, ...) \ + do { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #__VA_ARGS__); \ + DOCTEST_WRAP_IN_TRY( \ + _DOCTEST_RB.binary_assert( \ + __VA_ARGS__)) \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } while((void)0, 0) + +#define DOCTEST_UNARY_ASSERT(assert_type, ...) \ + do { \ + doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \ + __LINE__, #__VA_ARGS__); \ + DOCTEST_WRAP_IN_TRY(_DOCTEST_RB.unary_assert(__VA_ARGS__)) \ + DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB); \ + } while((void)0, 0) + +#else // DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_BINARY_ASSERT(assert_type, comparison, ...) \ + doctest::detail::binary_assert( \ + doctest::assertType::assert_type, __FILE__, __LINE__, #__VA_ARGS__, __VA_ARGS__) + +#define DOCTEST_UNARY_ASSERT(assert_type, ...) \ + doctest::detail::unary_assert(doctest::assertType::assert_type, __FILE__, __LINE__, \ + #__VA_ARGS__, __VA_ARGS__) + +#endif // DOCTEST_CONFIG_SUPER_FAST_ASSERTS + +#define DOCTEST_WARN_EQ(...) DOCTEST_BINARY_ASSERT(DT_WARN_EQ, eq, __VA_ARGS__) +#define DOCTEST_CHECK_EQ(...) DOCTEST_BINARY_ASSERT(DT_CHECK_EQ, eq, __VA_ARGS__) +#define DOCTEST_REQUIRE_EQ(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_EQ, eq, __VA_ARGS__) +#define DOCTEST_WARN_NE(...) DOCTEST_BINARY_ASSERT(DT_WARN_NE, ne, __VA_ARGS__) +#define DOCTEST_CHECK_NE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_NE, ne, __VA_ARGS__) +#define DOCTEST_REQUIRE_NE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_NE, ne, __VA_ARGS__) +#define DOCTEST_WARN_GT(...) DOCTEST_BINARY_ASSERT(DT_WARN_GT, gt, __VA_ARGS__) +#define DOCTEST_CHECK_GT(...) DOCTEST_BINARY_ASSERT(DT_CHECK_GT, gt, __VA_ARGS__) +#define DOCTEST_REQUIRE_GT(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_GT, gt, __VA_ARGS__) +#define DOCTEST_WARN_LT(...) DOCTEST_BINARY_ASSERT(DT_WARN_LT, lt, __VA_ARGS__) +#define DOCTEST_CHECK_LT(...) DOCTEST_BINARY_ASSERT(DT_CHECK_LT, lt, __VA_ARGS__) +#define DOCTEST_REQUIRE_LT(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_LT, lt, __VA_ARGS__) +#define DOCTEST_WARN_GE(...) DOCTEST_BINARY_ASSERT(DT_WARN_GE, ge, __VA_ARGS__) +#define DOCTEST_CHECK_GE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_GE, ge, __VA_ARGS__) +#define DOCTEST_REQUIRE_GE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_GE, ge, __VA_ARGS__) +#define DOCTEST_WARN_LE(...) DOCTEST_BINARY_ASSERT(DT_WARN_LE, le, __VA_ARGS__) +#define DOCTEST_CHECK_LE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_LE, le, __VA_ARGS__) +#define DOCTEST_REQUIRE_LE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_LE, le, __VA_ARGS__) + +#define DOCTEST_WARN_UNARY(...) DOCTEST_UNARY_ASSERT(DT_WARN_UNARY, __VA_ARGS__) +#define DOCTEST_CHECK_UNARY(...) DOCTEST_UNARY_ASSERT(DT_CHECK_UNARY, __VA_ARGS__) +#define DOCTEST_REQUIRE_UNARY(...) DOCTEST_UNARY_ASSERT(DT_REQUIRE_UNARY, __VA_ARGS__) +#define DOCTEST_WARN_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_WARN_UNARY_FALSE, __VA_ARGS__) +#define DOCTEST_CHECK_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_CHECK_UNARY_FALSE, __VA_ARGS__) +#define DOCTEST_REQUIRE_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_REQUIRE_UNARY_FALSE, __VA_ARGS__) + +#ifdef DOCTEST_CONFIG_NO_EXCEPTIONS + +#undef DOCTEST_WARN_THROWS +#undef DOCTEST_CHECK_THROWS +#undef DOCTEST_REQUIRE_THROWS +#undef DOCTEST_WARN_THROWS_AS +#undef DOCTEST_CHECK_THROWS_AS +#undef DOCTEST_REQUIRE_THROWS_AS +#undef DOCTEST_WARN_THROWS_WITH +#undef DOCTEST_CHECK_THROWS_WITH +#undef DOCTEST_REQUIRE_THROWS_WITH +#undef DOCTEST_WARN_NOTHROW +#undef DOCTEST_CHECK_NOTHROW +#undef DOCTEST_REQUIRE_NOTHROW + +#undef DOCTEST_WARN_THROWS_MESSAGE +#undef DOCTEST_CHECK_THROWS_MESSAGE +#undef DOCTEST_REQUIRE_THROWS_MESSAGE +#undef DOCTEST_WARN_THROWS_AS_MESSAGE +#undef DOCTEST_CHECK_THROWS_AS_MESSAGE +#undef DOCTEST_REQUIRE_THROWS_AS_MESSAGE +#undef DOCTEST_WARN_THROWS_WITH_MESSAGE +#undef DOCTEST_CHECK_THROWS_WITH_MESSAGE +#undef DOCTEST_REQUIRE_THROWS_WITH_MESSAGE +#undef DOCTEST_WARN_NOTHROW_MESSAGE +#undef DOCTEST_CHECK_NOTHROW_MESSAGE +#undef DOCTEST_REQUIRE_NOTHROW_MESSAGE + +#ifdef DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS + +#define DOCTEST_WARN_THROWS(expr) ((void)0) +#define DOCTEST_CHECK_THROWS(expr) ((void)0) +#define DOCTEST_REQUIRE_THROWS(expr) ((void)0) +#define DOCTEST_WARN_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_CHECK_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_REQUIRE_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_WARN_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_CHECK_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_WARN_NOTHROW(expr) ((void)0) +#define DOCTEST_CHECK_NOTHROW(expr) ((void)0) +#define DOCTEST_REQUIRE_NOTHROW(expr) ((void)0) + +#define DOCTEST_WARN_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_WARN_NOTHROW_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, msg) ((void)0) + +#else // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS + +#undef DOCTEST_REQUIRE +#undef DOCTEST_REQUIRE_FALSE +#undef DOCTEST_REQUIRE_MESSAGE +#undef DOCTEST_REQUIRE_FALSE_MESSAGE +#undef DOCTEST_REQUIRE_EQ +#undef DOCTEST_REQUIRE_NE +#undef DOCTEST_REQUIRE_GT +#undef DOCTEST_REQUIRE_LT +#undef DOCTEST_REQUIRE_GE +#undef DOCTEST_REQUIRE_LE +#undef DOCTEST_REQUIRE_UNARY +#undef DOCTEST_REQUIRE_UNARY_FALSE + +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS + +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + +// ================================================================================================= +// == WHAT FOLLOWS IS VERSIONS OF THE MACROS THAT DO NOT DO ANY REGISTERING! == +// == THIS CAN BE ENABLED BY DEFINING DOCTEST_CONFIG_DISABLE GLOBALLY! == +// ================================================================================================= +#else // DOCTEST_CONFIG_DISABLE + +#define DOCTEST_IMPLEMENT_FIXTURE(der, base, func, name) \ + namespace { \ + template \ + struct der : public base \ + { void f(); }; \ + } \ + template \ + inline void der::f() + +#define DOCTEST_CREATE_AND_REGISTER_FUNCTION(f, name) \ + template \ + static inline void f() + +// for registering tests +#define DOCTEST_TEST_CASE(name) \ + DOCTEST_CREATE_AND_REGISTER_FUNCTION(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), name) + +// for registering tests with a fixture +#define DOCTEST_TEST_CASE_FIXTURE(x, name) \ + DOCTEST_IMPLEMENT_FIXTURE(DOCTEST_ANONYMOUS(_DOCTEST_ANON_CLASS_), x, \ + DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), name) + +// for converting types to strings without the header and demangling +#define DOCTEST_TYPE_TO_STRING(...) typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) +#define DOCTEST_TYPE_TO_STRING_IMPL(...) + +// for typed tests +#define DOCTEST_TEST_CASE_TEMPLATE(name, type, ...) \ + template \ + inline void DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_)() + +#define DOCTEST_TEST_CASE_TEMPLATE_DEFINE(name, type, id) \ + template \ + inline void DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_)() + +#define DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE(id, ...) \ + typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for subcases +#define DOCTEST_SUBCASE(name) + +// for a testsuite block +#define DOCTEST_TEST_SUITE(name) namespace + +// for starting a testsuite block +#define DOCTEST_TEST_SUITE_BEGIN(name) typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +// for ending a testsuite block +#define DOCTEST_TEST_SUITE_END typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_) + +#define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR(signature) \ + template \ + static inline doctest::String DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_)(signature) + +#define DOCTEST_REGISTER_REPORTER(name, priority, reporter) + +#define DOCTEST_INFO(x) ((void)0) +#define DOCTEST_CAPTURE(x) ((void)0) +#define DOCTEST_ADD_MESSAGE_AT(file, line, x) ((void)0) +#define DOCTEST_ADD_FAIL_CHECK_AT(file, line, x) ((void)0) +#define DOCTEST_ADD_FAIL_AT(file, line, x) ((void)0) +#define DOCTEST_MESSAGE(x) ((void)0) +#define DOCTEST_FAIL_CHECK(x) ((void)0) +#define DOCTEST_FAIL(x) ((void)0) + +#define DOCTEST_WARN(...) ((void)0) +#define DOCTEST_CHECK(...) ((void)0) +#define DOCTEST_REQUIRE(...) ((void)0) +#define DOCTEST_WARN_FALSE(...) ((void)0) +#define DOCTEST_CHECK_FALSE(...) ((void)0) +#define DOCTEST_REQUIRE_FALSE(...) ((void)0) + +#define DOCTEST_WARN_MESSAGE(cond, msg) ((void)0) +#define DOCTEST_CHECK_MESSAGE(cond, msg) ((void)0) +#define DOCTEST_REQUIRE_MESSAGE(cond, msg) ((void)0) +#define DOCTEST_WARN_FALSE_MESSAGE(cond, msg) ((void)0) +#define DOCTEST_CHECK_FALSE_MESSAGE(cond, msg) ((void)0) +#define DOCTEST_REQUIRE_FALSE_MESSAGE(cond, msg) ((void)0) + +#define DOCTEST_WARN_THROWS(expr) ((void)0) +#define DOCTEST_CHECK_THROWS(expr) ((void)0) +#define DOCTEST_REQUIRE_THROWS(expr) ((void)0) +#define DOCTEST_WARN_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_CHECK_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_REQUIRE_THROWS_AS(expr, ...) ((void)0) +#define DOCTEST_WARN_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_CHECK_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) ((void)0) +#define DOCTEST_WARN_NOTHROW(expr) ((void)0) +#define DOCTEST_CHECK_NOTHROW(expr) ((void)0) +#define DOCTEST_REQUIRE_NOTHROW(expr) ((void)0) + +#define DOCTEST_WARN_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, ex, msg) ((void)0) +#define DOCTEST_WARN_NOTHROW_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, msg) ((void)0) +#define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, msg) ((void)0) + +#define DOCTEST_WARN_EQ(...) ((void)0) +#define DOCTEST_CHECK_EQ(...) ((void)0) +#define DOCTEST_REQUIRE_EQ(...) ((void)0) +#define DOCTEST_WARN_NE(...) ((void)0) +#define DOCTEST_CHECK_NE(...) ((void)0) +#define DOCTEST_REQUIRE_NE(...) ((void)0) +#define DOCTEST_WARN_GT(...) ((void)0) +#define DOCTEST_CHECK_GT(...) ((void)0) +#define DOCTEST_REQUIRE_GT(...) ((void)0) +#define DOCTEST_WARN_LT(...) ((void)0) +#define DOCTEST_CHECK_LT(...) ((void)0) +#define DOCTEST_REQUIRE_LT(...) ((void)0) +#define DOCTEST_WARN_GE(...) ((void)0) +#define DOCTEST_CHECK_GE(...) ((void)0) +#define DOCTEST_REQUIRE_GE(...) ((void)0) +#define DOCTEST_WARN_LE(...) ((void)0) +#define DOCTEST_CHECK_LE(...) ((void)0) +#define DOCTEST_REQUIRE_LE(...) ((void)0) + +#define DOCTEST_WARN_UNARY(...) ((void)0) +#define DOCTEST_CHECK_UNARY(...) ((void)0) +#define DOCTEST_REQUIRE_UNARY(...) ((void)0) +#define DOCTEST_WARN_UNARY_FALSE(...) ((void)0) +#define DOCTEST_CHECK_UNARY_FALSE(...) ((void)0) +#define DOCTEST_REQUIRE_UNARY_FALSE(...) ((void)0) + +#endif // DOCTEST_CONFIG_DISABLE + +// clang-format off +// KEPT FOR BACKWARDS COMPATIBILITY - FORWARDING TO THE RIGHT MACROS +#define DOCTEST_FAST_WARN_EQ DOCTEST_WARN_EQ +#define DOCTEST_FAST_CHECK_EQ DOCTEST_CHECK_EQ +#define DOCTEST_FAST_REQUIRE_EQ DOCTEST_REQUIRE_EQ +#define DOCTEST_FAST_WARN_NE DOCTEST_WARN_NE +#define DOCTEST_FAST_CHECK_NE DOCTEST_CHECK_NE +#define DOCTEST_FAST_REQUIRE_NE DOCTEST_REQUIRE_NE +#define DOCTEST_FAST_WARN_GT DOCTEST_WARN_GT +#define DOCTEST_FAST_CHECK_GT DOCTEST_CHECK_GT +#define DOCTEST_FAST_REQUIRE_GT DOCTEST_REQUIRE_GT +#define DOCTEST_FAST_WARN_LT DOCTEST_WARN_LT +#define DOCTEST_FAST_CHECK_LT DOCTEST_CHECK_LT +#define DOCTEST_FAST_REQUIRE_LT DOCTEST_REQUIRE_LT +#define DOCTEST_FAST_WARN_GE DOCTEST_WARN_GE +#define DOCTEST_FAST_CHECK_GE DOCTEST_CHECK_GE +#define DOCTEST_FAST_REQUIRE_GE DOCTEST_REQUIRE_GE +#define DOCTEST_FAST_WARN_LE DOCTEST_WARN_LE +#define DOCTEST_FAST_CHECK_LE DOCTEST_CHECK_LE +#define DOCTEST_FAST_REQUIRE_LE DOCTEST_REQUIRE_LE + +#define DOCTEST_FAST_WARN_UNARY DOCTEST_WARN_UNARY +#define DOCTEST_FAST_CHECK_UNARY DOCTEST_CHECK_UNARY +#define DOCTEST_FAST_REQUIRE_UNARY DOCTEST_REQUIRE_UNARY +#define DOCTEST_FAST_WARN_UNARY_FALSE DOCTEST_WARN_UNARY_FALSE +#define DOCTEST_FAST_CHECK_UNARY_FALSE DOCTEST_CHECK_UNARY_FALSE +#define DOCTEST_FAST_REQUIRE_UNARY_FALSE DOCTEST_REQUIRE_UNARY_FALSE +// clang-format on + +// BDD style macros +// clang-format off +#define DOCTEST_SCENARIO(name) DOCTEST_TEST_CASE(" Scenario: " name) +#define DOCTEST_SCENARIO_TEMPLATE(name, T, ...) DOCTEST_TEST_CASE_TEMPLATE(" Scenario: " name, T, __VA_ARGS__) +#define DOCTEST_SCENARIO_TEMPLATE_DEFINE(name, T, id) DOCTEST_TEST_CASE_TEMPLATE_DEFINE(" Scenario: " name, T, id) + +#define DOCTEST_GIVEN(name) SUBCASE(" Given: " name) +#define DOCTEST_WHEN(name) SUBCASE(" When: " name) +#define DOCTEST_AND_WHEN(name) SUBCASE("And when: " name) +#define DOCTEST_THEN(name) SUBCASE(" Then: " name) +#define DOCTEST_AND_THEN(name) SUBCASE(" And: " name) +// clang-format on + +// == SHORT VERSIONS OF THE MACROS +#if !defined(DOCTEST_CONFIG_NO_SHORT_MACRO_NAMES) + +#define TEST_CASE DOCTEST_TEST_CASE +#define TEST_CASE_FIXTURE DOCTEST_TEST_CASE_FIXTURE +#define TYPE_TO_STRING DOCTEST_TYPE_TO_STRING +#define TEST_CASE_TEMPLATE DOCTEST_TEST_CASE_TEMPLATE +#define TEST_CASE_TEMPLATE_DEFINE DOCTEST_TEST_CASE_TEMPLATE_DEFINE +#define TEST_CASE_TEMPLATE_INSTANTIATE DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE +#define SUBCASE DOCTEST_SUBCASE +#define TEST_SUITE DOCTEST_TEST_SUITE +#define TEST_SUITE_BEGIN DOCTEST_TEST_SUITE_BEGIN +#define TEST_SUITE_END DOCTEST_TEST_SUITE_END +#define REGISTER_EXCEPTION_TRANSLATOR DOCTEST_REGISTER_EXCEPTION_TRANSLATOR +#define REGISTER_REPORTER DOCTEST_REGISTER_REPORTER +#define INFO DOCTEST_INFO +#define CAPTURE DOCTEST_CAPTURE +#define ADD_MESSAGE_AT DOCTEST_ADD_MESSAGE_AT +#define ADD_FAIL_CHECK_AT DOCTEST_ADD_FAIL_CHECK_AT +#define ADD_FAIL_AT DOCTEST_ADD_FAIL_AT +#define MESSAGE DOCTEST_MESSAGE +#define FAIL_CHECK DOCTEST_FAIL_CHECK +#define FAIL DOCTEST_FAIL +#define TO_LVALUE DOCTEST_TO_LVALUE + +#define WARN DOCTEST_WARN +#define WARN_FALSE DOCTEST_WARN_FALSE +#define WARN_THROWS DOCTEST_WARN_THROWS +#define WARN_THROWS_AS DOCTEST_WARN_THROWS_AS +#define WARN_THROWS_WITH DOCTEST_WARN_THROWS_WITH +#define WARN_NOTHROW DOCTEST_WARN_NOTHROW +#define CHECK DOCTEST_CHECK +#define CHECK_FALSE DOCTEST_CHECK_FALSE +#define CHECK_THROWS DOCTEST_CHECK_THROWS +#define CHECK_THROWS_AS DOCTEST_CHECK_THROWS_AS +#define CHECK_THROWS_WITH DOCTEST_CHECK_THROWS_WITH +#define CHECK_NOTHROW DOCTEST_CHECK_NOTHROW +#define REQUIRE DOCTEST_REQUIRE +#define REQUIRE_FALSE DOCTEST_REQUIRE_FALSE +#define REQUIRE_THROWS DOCTEST_REQUIRE_THROWS +#define REQUIRE_THROWS_AS DOCTEST_REQUIRE_THROWS_AS +#define REQUIRE_THROWS_WITH DOCTEST_REQUIRE_THROWS_WITH +#define REQUIRE_NOTHROW DOCTEST_REQUIRE_NOTHROW + +#define WARN_MESSAGE DOCTEST_WARN_MESSAGE +#define WARN_FALSE_MESSAGE DOCTEST_WARN_FALSE_MESSAGE +#define WARN_THROWS_MESSAGE DOCTEST_WARN_THROWS_MESSAGE +#define WARN_THROWS_AS_MESSAGE DOCTEST_WARN_THROWS_AS_MESSAGE +#define WARN_THROWS_WITH_MESSAGE DOCTEST_WARN_THROWS_WITH_MESSAGE +#define WARN_NOTHROW_MESSAGE DOCTEST_WARN_NOTHROW_MESSAGE +#define CHECK_MESSAGE DOCTEST_CHECK_MESSAGE +#define CHECK_FALSE_MESSAGE DOCTEST_CHECK_FALSE_MESSAGE +#define CHECK_THROWS_MESSAGE DOCTEST_CHECK_THROWS_MESSAGE +#define CHECK_THROWS_AS_MESSAGE DOCTEST_CHECK_THROWS_AS_MESSAGE +#define CHECK_THROWS_WITH_MESSAGE DOCTEST_CHECK_THROWS_WITH_MESSAGE +#define CHECK_NOTHROW_MESSAGE DOCTEST_CHECK_NOTHROW_MESSAGE +#define REQUIRE_MESSAGE DOCTEST_REQUIRE_MESSAGE +#define REQUIRE_FALSE_MESSAGE DOCTEST_REQUIRE_FALSE_MESSAGE +#define REQUIRE_THROWS_MESSAGE DOCTEST_REQUIRE_THROWS_MESSAGE +#define REQUIRE_THROWS_AS_MESSAGE DOCTEST_REQUIRE_THROWS_AS_MESSAGE +#define REQUIRE_THROWS_WITH_MESSAGE DOCTEST_REQUIRE_THROWS_WITH_MESSAGE +#define REQUIRE_NOTHROW_MESSAGE DOCTEST_REQUIRE_NOTHROW_MESSAGE + +#define SCENARIO DOCTEST_SCENARIO +#define SCENARIO_TEMPLATE DOCTEST_SCENARIO_TEMPLATE +#define SCENARIO_TEMPLATE_DEFINE DOCTEST_SCENARIO_TEMPLATE_DEFINE +#define GIVEN DOCTEST_GIVEN +#define WHEN DOCTEST_WHEN +#define AND_WHEN DOCTEST_AND_WHEN +#define THEN DOCTEST_THEN +#define AND_THEN DOCTEST_AND_THEN + +#define WARN_EQ DOCTEST_WARN_EQ +#define CHECK_EQ DOCTEST_CHECK_EQ +#define REQUIRE_EQ DOCTEST_REQUIRE_EQ +#define WARN_NE DOCTEST_WARN_NE +#define CHECK_NE DOCTEST_CHECK_NE +#define REQUIRE_NE DOCTEST_REQUIRE_NE +#define WARN_GT DOCTEST_WARN_GT +#define CHECK_GT DOCTEST_CHECK_GT +#define REQUIRE_GT DOCTEST_REQUIRE_GT +#define WARN_LT DOCTEST_WARN_LT +#define CHECK_LT DOCTEST_CHECK_LT +#define REQUIRE_LT DOCTEST_REQUIRE_LT +#define WARN_GE DOCTEST_WARN_GE +#define CHECK_GE DOCTEST_CHECK_GE +#define REQUIRE_GE DOCTEST_REQUIRE_GE +#define WARN_LE DOCTEST_WARN_LE +#define CHECK_LE DOCTEST_CHECK_LE +#define REQUIRE_LE DOCTEST_REQUIRE_LE +#define WARN_UNARY DOCTEST_WARN_UNARY +#define CHECK_UNARY DOCTEST_CHECK_UNARY +#define REQUIRE_UNARY DOCTEST_REQUIRE_UNARY +#define WARN_UNARY_FALSE DOCTEST_WARN_UNARY_FALSE +#define CHECK_UNARY_FALSE DOCTEST_CHECK_UNARY_FALSE +#define REQUIRE_UNARY_FALSE DOCTEST_REQUIRE_UNARY_FALSE + +// KEPT FOR BACKWARDS COMPATIBILITY +#define FAST_WARN_EQ DOCTEST_FAST_WARN_EQ +#define FAST_CHECK_EQ DOCTEST_FAST_CHECK_EQ +#define FAST_REQUIRE_EQ DOCTEST_FAST_REQUIRE_EQ +#define FAST_WARN_NE DOCTEST_FAST_WARN_NE +#define FAST_CHECK_NE DOCTEST_FAST_CHECK_NE +#define FAST_REQUIRE_NE DOCTEST_FAST_REQUIRE_NE +#define FAST_WARN_GT DOCTEST_FAST_WARN_GT +#define FAST_CHECK_GT DOCTEST_FAST_CHECK_GT +#define FAST_REQUIRE_GT DOCTEST_FAST_REQUIRE_GT +#define FAST_WARN_LT DOCTEST_FAST_WARN_LT +#define FAST_CHECK_LT DOCTEST_FAST_CHECK_LT +#define FAST_REQUIRE_LT DOCTEST_FAST_REQUIRE_LT +#define FAST_WARN_GE DOCTEST_FAST_WARN_GE +#define FAST_CHECK_GE DOCTEST_FAST_CHECK_GE +#define FAST_REQUIRE_GE DOCTEST_FAST_REQUIRE_GE +#define FAST_WARN_LE DOCTEST_FAST_WARN_LE +#define FAST_CHECK_LE DOCTEST_FAST_CHECK_LE +#define FAST_REQUIRE_LE DOCTEST_FAST_REQUIRE_LE +#define FAST_WARN_UNARY DOCTEST_FAST_WARN_UNARY +#define FAST_CHECK_UNARY DOCTEST_FAST_CHECK_UNARY +#define FAST_REQUIRE_UNARY DOCTEST_FAST_REQUIRE_UNARY +#define FAST_WARN_UNARY_FALSE DOCTEST_FAST_WARN_UNARY_FALSE +#define FAST_CHECK_UNARY_FALSE DOCTEST_FAST_CHECK_UNARY_FALSE +#define FAST_REQUIRE_UNARY_FALSE DOCTEST_FAST_REQUIRE_UNARY_FALSE + +#endif // DOCTEST_CONFIG_NO_SHORT_MACRO_NAMES + +#if !defined(DOCTEST_CONFIG_DISABLE) + +// this is here to clear the 'current test suite' for the current translation unit - at the top +DOCTEST_TEST_SUITE_END(); + +// add stringification for primitive/fundamental types +namespace doctest { namespace detail { + DOCTEST_TYPE_TO_STRING_IMPL(bool) + DOCTEST_TYPE_TO_STRING_IMPL(float) + DOCTEST_TYPE_TO_STRING_IMPL(double) + DOCTEST_TYPE_TO_STRING_IMPL(long double) + DOCTEST_TYPE_TO_STRING_IMPL(char) + DOCTEST_TYPE_TO_STRING_IMPL(signed char) + DOCTEST_TYPE_TO_STRING_IMPL(unsigned char) + DOCTEST_TYPE_TO_STRING_IMPL(wchar_t) + DOCTEST_TYPE_TO_STRING_IMPL(short int) + DOCTEST_TYPE_TO_STRING_IMPL(unsigned short int) + DOCTEST_TYPE_TO_STRING_IMPL(int) + DOCTEST_TYPE_TO_STRING_IMPL(unsigned int) + DOCTEST_TYPE_TO_STRING_IMPL(long int) + DOCTEST_TYPE_TO_STRING_IMPL(unsigned long int) + DOCTEST_TYPE_TO_STRING_IMPL(long long int) + DOCTEST_TYPE_TO_STRING_IMPL(unsigned long long int) +}} // namespace doctest::detail + +#endif // DOCTEST_CONFIG_DISABLE + +DOCTEST_CLANG_SUPPRESS_WARNING_POP +DOCTEST_MSVC_SUPPRESS_WARNING_POP +DOCTEST_GCC_SUPPRESS_WARNING_POP + +#endif // DOCTEST_LIBRARY_INCLUDED + +#ifndef DOCTEST_SINGLE_HEADER +#define DOCTEST_SINGLE_HEADER +#endif // DOCTEST_SINGLE_HEADER + +#if defined(DOCTEST_CONFIG_IMPLEMENT) || !defined(DOCTEST_SINGLE_HEADER) +#ifndef DOCTEST_LIBRARY_IMPLEMENTATION +#define DOCTEST_LIBRARY_IMPLEMENTATION + +#ifndef DOCTEST_SINGLE_HEADER +#include "doctest_fwd.h" +#endif // DOCTEST_SINGLE_HEADER + +DOCTEST_CLANG_SUPPRESS_WARNING_PUSH +DOCTEST_CLANG_SUPPRESS_WARNING("-Wunknown-pragmas") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wpadded") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wweak-vtables") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wglobal-constructors") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wexit-time-destructors") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-prototypes") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-conversion") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wshorten-64-to-32") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-variable-declarations") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wswitch") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wswitch-enum") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wcovered-switch-default") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-noreturn") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-local-typedef") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wdisabled-macro-expansion") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-braces") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-field-initializers") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat") +DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat-pedantic") + +DOCTEST_GCC_SUPPRESS_WARNING_PUSH +DOCTEST_GCC_SUPPRESS_WARNING("-Wunknown-pragmas") +DOCTEST_GCC_SUPPRESS_WARNING("-Wpragmas") +DOCTEST_GCC_SUPPRESS_WARNING("-Wconversion") +DOCTEST_GCC_SUPPRESS_WARNING("-Weffc++") +DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-conversion") +DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-overflow") +DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-aliasing") +DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-field-initializers") +DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-braces") +DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-declarations") +DOCTEST_GCC_SUPPRESS_WARNING("-Winline") +DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch") +DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch-enum") +DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch-default") +DOCTEST_GCC_SUPPRESS_WARNING("-Wunsafe-loop-optimizations") +DOCTEST_GCC_SUPPRESS_WARNING("-Wold-style-cast") +DOCTEST_GCC_SUPPRESS_WARNING("-Wunused-local-typedefs") +DOCTEST_GCC_SUPPRESS_WARNING("-Wuseless-cast") + +DOCTEST_MSVC_SUPPRESS_WARNING_PUSH +DOCTEST_MSVC_SUPPRESS_WARNING(4616) // invalid compiler warning +DOCTEST_MSVC_SUPPRESS_WARNING(4619) // invalid compiler warning +DOCTEST_MSVC_SUPPRESS_WARNING(4996) // The compiler encountered a deprecated declaration +DOCTEST_MSVC_SUPPRESS_WARNING(4267) // 'var' : conversion from 'x' to 'y', possible loss of data +DOCTEST_MSVC_SUPPRESS_WARNING(4706) // assignment within conditional expression +DOCTEST_MSVC_SUPPRESS_WARNING(4512) // 'class' : assignment operator could not be generated +DOCTEST_MSVC_SUPPRESS_WARNING(4127) // conditional expression is constant +DOCTEST_MSVC_SUPPRESS_WARNING(4530) // C++ exception handler used, but unwind semantics not enabled +DOCTEST_MSVC_SUPPRESS_WARNING(4577) // 'noexcept' used with no exception handling mode specified +DOCTEST_MSVC_SUPPRESS_WARNING(4774) // format string expected in argument is not a string literal +DOCTEST_MSVC_SUPPRESS_WARNING(4365) // conversion from 'int' to 'unsigned', signed/unsigned mismatch +DOCTEST_MSVC_SUPPRESS_WARNING(4820) // padding in structs +DOCTEST_MSVC_SUPPRESS_WARNING(4640) // construction of local static object is not thread-safe +DOCTEST_MSVC_SUPPRESS_WARNING(5039) // pointer to potentially throwing function passed to extern C +DOCTEST_MSVC_SUPPRESS_WARNING(5045) // Spectre mitigation stuff +DOCTEST_MSVC_SUPPRESS_WARNING(4626) // assignment operator was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(5027) // move assignment operator was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(5026) // move constructor was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(4625) // copy constructor was implicitly defined as deleted +DOCTEST_MSVC_SUPPRESS_WARNING(4800) // forcing value to bool 'true' or 'false' (performance warning) +// static analysis +DOCTEST_MSVC_SUPPRESS_WARNING(26439) // This kind of function may not throw. Declare it 'noexcept' +DOCTEST_MSVC_SUPPRESS_WARNING(26495) // Always initialize a member variable +DOCTEST_MSVC_SUPPRESS_WARNING(26451) // Arithmetic overflow ... +DOCTEST_MSVC_SUPPRESS_WARNING(26444) // Avoid unnamed objects with custom construction and dtor... + +DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_BEGIN + +// required includes - will go only in one translation unit! +#include +#include +#include +// borland (Embarcadero) compiler requires math.h and not cmath - https://github.com/onqtam/doctest/pull/37 +#ifdef __BORLANDC__ +#include +#endif // __BORLANDC__ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#ifdef DOCTEST_PLATFORM_MAC +#include +#include +#include +#endif // DOCTEST_PLATFORM_MAC + +DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_END + +// counts the number of elements in a C array +#define DOCTEST_COUNTOF(x) (sizeof(x) / sizeof(x[0])) + +#ifdef DOCTEST_CONFIG_DISABLE +#define DOCTEST_BRANCH_ON_DISABLED(if_disabled, if_not_disabled) if_disabled +#else // DOCTEST_CONFIG_DISABLE +#define DOCTEST_BRANCH_ON_DISABLED(if_disabled, if_not_disabled) if_not_disabled +#endif // DOCTEST_CONFIG_DISABLE + +#ifndef DOCTEST_CONFIG_OPTIONS_PREFIX +#define DOCTEST_CONFIG_OPTIONS_PREFIX "dt-" +#endif + +#ifndef DOCTEST_THREAD_LOCAL +#define DOCTEST_THREAD_LOCAL thread_local +#endif + +#ifdef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS +#define DOCTEST_OPTIONS_PREFIX_DISPLAY DOCTEST_CONFIG_OPTIONS_PREFIX +#else +#define DOCTEST_OPTIONS_PREFIX_DISPLAY "" +#endif + +namespace doctest { + +bool is_running_in_test = false; + +namespace { + using namespace detail; + // case insensitive strcmp + int stricmp(const char* a, const char* b) { + for(;; a++, b++) { + const int d = tolower(*a) - tolower(*b); + if(d != 0 || !*a) + return d; + } + } + + template + String fpToString(T value, int precision) { + std::ostringstream oss; + oss << std::setprecision(precision) << std::fixed << value; + std::string d = oss.str(); + size_t i = d.find_last_not_of('0'); + if(i != std::string::npos && i != d.size() - 1) { + if(d[i] == '.') + i++; + d = d.substr(0, i + 1); + } + return d.c_str(); + } + + struct Endianness + { + enum Arch + { + Big, + Little + }; + + static Arch which() { + union _ + { + int asInt; + char asChar[sizeof(int)]; + } u; + + u.asInt = 1; // NOLINT + return (u.asChar[sizeof(int) - 1] == 1) ? Big : Little; // NOLINT + } + }; +} // namespace + +namespace detail { + void my_memcpy(void* dest, const void* src, unsigned num) { memcpy(dest, src, num); } + + String rawMemoryToString(const void* object, unsigned size) { + // Reverse order for little endian architectures + int i = 0, end = static_cast(size), inc = 1; + if(Endianness::which() == Endianness::Little) { + i = end - 1; + end = inc = -1; + } + + unsigned const char* bytes = static_cast(object); + std::ostringstream oss; + oss << "0x" << std::setfill('0') << std::hex; + for(; i != end; i += inc) + oss << std::setw(2) << static_cast(bytes[i]); + return oss.str().c_str(); + } + + DOCTEST_THREAD_LOCAL std::ostringstream g_oss; + + std::ostream* getTlsOss() { + g_oss.clear(); // there shouldn't be anything worth clearing in the flags + g_oss.str(""); // the slow way of resetting a string stream + //g_oss.seekp(0); // optimal reset - as seen here: https://stackoverflow.com/a/624291/3162383 + return &g_oss; + } + + String getTlsOssResult() { + //g_oss << std::ends; // needed - as shown here: https://stackoverflow.com/a/624291/3162383 + return g_oss.str().c_str(); + } + +#ifndef DOCTEST_CONFIG_DISABLE + // this holds both parameters from the command line and runtime data for tests + struct ContextState : ContextOptions, TestRunStats, CurrentTestCaseStats + { + std::atomic numAssertsForCurrentTestCase_atomic; + std::atomic numAssertsFailedForCurrentTestCase_atomic; + + std::vector > filters = decltype(filters)(9); // 9 different filters + + std::vector reporters_currently_used; + + const TestCase* currentTest = nullptr; + + assert_handler ah = nullptr; + + std::vector stringifiedContexts; // logging from INFO() due to an exception + + // stuff for subcases + std::set subcasesPassed; + std::set subcasesEnteredLevels; + int subcasesCurrentLevel; + + void resetRunData() { + numTestCases = 0; + numTestCasesPassingFilters = 0; + numTestSuitesPassingFilters = 0; + numTestCasesFailed = 0; + numAsserts = 0; + numAssertsFailed = 0; + } + }; + + ContextState* g_cs = nullptr; + DOCTEST_THREAD_LOCAL bool g_no_colors; // used to avoid locks for the debug output + +#endif // DOCTEST_CONFIG_DISABLE +} // namespace detail + +void String::setOnHeap() { *reinterpret_cast(&buf[last]) = 128; } +void String::setLast(unsigned in) { buf[last] = char(in); } + +void String::copy(const String& other) { + if(other.isOnStack()) { + memcpy(buf, other.buf, len); + } else { + setOnHeap(); + data.size = other.data.size; + data.capacity = data.size + 1; + data.ptr = new char[data.capacity]; + memcpy(data.ptr, other.data.ptr, data.size + 1); + } +} + +String::String() { + buf[0] = '\0'; + setLast(); +} + +String::~String() { + if(!isOnStack()) + delete[] data.ptr; +} + +String::String(const char* in) + : String(in, strlen(in)) {} + +String::String(const char* in, unsigned in_size) { + if(in_size <= last) { + memcpy(buf, in, in_size + 1); + setLast(last - in_size); + } else { + setOnHeap(); + data.size = in_size; + data.capacity = data.size + 1; + data.ptr = new char[data.capacity]; + memcpy(data.ptr, in, in_size + 1); + } +} + +String::String(const String& other) { copy(other); } + +String& String::operator=(const String& other) { + if(this != &other) { + if(!isOnStack()) + delete[] data.ptr; + + copy(other); + } + + return *this; +} + +String& String::operator+=(const String& other) { + const unsigned my_old_size = size(); + const unsigned other_size = other.size(); + const unsigned total_size = my_old_size + other_size; + if(isOnStack()) { + if(total_size < len) { + // append to the current stack space + memcpy(buf + my_old_size, other.c_str(), other_size + 1); + setLast(last - total_size); + } else { + // alloc new chunk + char* temp = new char[total_size + 1]; + // copy current data to new location before writing in the union + memcpy(temp, buf, my_old_size); // skip the +1 ('\0') for speed + // update data in union + setOnHeap(); + data.size = total_size; + data.capacity = data.size + 1; + data.ptr = temp; + // transfer the rest of the data + memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1); + } + } else { + if(data.capacity > total_size) { + // append to the current heap block + data.size = total_size; + memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1); + } else { + // resize + data.capacity *= 2; + if(data.capacity <= total_size) + data.capacity = total_size + 1; + // alloc new chunk + char* temp = new char[data.capacity]; + // copy current data to new location before releasing it + memcpy(temp, data.ptr, my_old_size); // skip the +1 ('\0') for speed + // release old chunk + delete[] data.ptr; + // update the rest of the union members + data.size = total_size; + data.ptr = temp; + // transfer the rest of the data + memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1); + } + } + + return *this; +} + +String String::operator+(const String& other) const { return String(*this) += other; } + +String::String(String&& other) { + memcpy(buf, other.buf, len); + other.buf[0] = '\0'; + other.setLast(); +} + +String& String::operator=(String&& other) { + if(this != &other) { + if(!isOnStack()) + delete[] data.ptr; + memcpy(buf, other.buf, len); + other.buf[0] = '\0'; + other.setLast(); + } + return *this; +} + +char String::operator[](unsigned i) const { + return const_cast(this)->operator[](i); // NOLINT +} + +char& String::operator[](unsigned i) { + if(isOnStack()) + return reinterpret_cast(buf)[i]; + return data.ptr[i]; +} + +DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wmaybe-uninitialized") +unsigned String::size() const { + if(isOnStack()) + return last - (unsigned(buf[last]) & 31); // using "last" would work only if "len" is 32 + return data.size; +} +DOCTEST_GCC_SUPPRESS_WARNING_POP + +unsigned String::capacity() const { + if(isOnStack()) + return len; + return data.capacity; +} + +int String::compare(const char* other, bool no_case) const { + if(no_case) + return stricmp(c_str(), other); + return std::strcmp(c_str(), other); +} + +int String::compare(const String& other, bool no_case) const { + return compare(other.c_str(), no_case); +} + +// clang-format off +bool operator==(const String& lhs, const String& rhs) { return lhs.compare(rhs) == 0; } +bool operator!=(const String& lhs, const String& rhs) { return lhs.compare(rhs) != 0; } +bool operator< (const String& lhs, const String& rhs) { return lhs.compare(rhs) < 0; } +bool operator> (const String& lhs, const String& rhs) { return lhs.compare(rhs) > 0; } +bool operator<=(const String& lhs, const String& rhs) { return (lhs != rhs) ? lhs.compare(rhs) < 0 : true; } +bool operator>=(const String& lhs, const String& rhs) { return (lhs != rhs) ? lhs.compare(rhs) > 0 : true; } +// clang-format on + +std::ostream& operator<<(std::ostream& s, const String& in) { return s << in.c_str(); } + +namespace { + void color_to_stream(std::ostream&, Color::Enum) DOCTEST_BRANCH_ON_DISABLED({}, ;) +} // namespace + +namespace Color { + std::ostream& operator<<(std::ostream& s, Color::Enum code) { + color_to_stream(s, code); + return s; + } +} // namespace Color + +// clang-format off +const char* assertString(assertType::Enum at) { + DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4062) // enum 'x' in switch of enum 'y' is not handled + switch(at) { //!OCLINT missing default in switch statements + case assertType::DT_WARN : return "WARN"; + case assertType::DT_CHECK : return "CHECK"; + case assertType::DT_REQUIRE : return "REQUIRE"; + + case assertType::DT_WARN_FALSE : return "WARN_FALSE"; + case assertType::DT_CHECK_FALSE : return "CHECK_FALSE"; + case assertType::DT_REQUIRE_FALSE : return "REQUIRE_FALSE"; + + case assertType::DT_WARN_THROWS : return "WARN_THROWS"; + case assertType::DT_CHECK_THROWS : return "CHECK_THROWS"; + case assertType::DT_REQUIRE_THROWS : return "REQUIRE_THROWS"; + + case assertType::DT_WARN_THROWS_AS : return "WARN_THROWS_AS"; + case assertType::DT_CHECK_THROWS_AS : return "CHECK_THROWS_AS"; + case assertType::DT_REQUIRE_THROWS_AS : return "REQUIRE_THROWS_AS"; + + case assertType::DT_WARN_THROWS_WITH : return "WARN_THROWS_WITH"; + case assertType::DT_CHECK_THROWS_WITH : return "CHECK_THROWS_WITH"; + case assertType::DT_REQUIRE_THROWS_WITH : return "REQUIRE_THROWS_WITH"; + + case assertType::DT_WARN_NOTHROW : return "WARN_NOTHROW"; + case assertType::DT_CHECK_NOTHROW : return "CHECK_NOTHROW"; + case assertType::DT_REQUIRE_NOTHROW : return "REQUIRE_NOTHROW"; + + case assertType::DT_WARN_EQ : return "WARN_EQ"; + case assertType::DT_CHECK_EQ : return "CHECK_EQ"; + case assertType::DT_REQUIRE_EQ : return "REQUIRE_EQ"; + case assertType::DT_WARN_NE : return "WARN_NE"; + case assertType::DT_CHECK_NE : return "CHECK_NE"; + case assertType::DT_REQUIRE_NE : return "REQUIRE_NE"; + case assertType::DT_WARN_GT : return "WARN_GT"; + case assertType::DT_CHECK_GT : return "CHECK_GT"; + case assertType::DT_REQUIRE_GT : return "REQUIRE_GT"; + case assertType::DT_WARN_LT : return "WARN_LT"; + case assertType::DT_CHECK_LT : return "CHECK_LT"; + case assertType::DT_REQUIRE_LT : return "REQUIRE_LT"; + case assertType::DT_WARN_GE : return "WARN_GE"; + case assertType::DT_CHECK_GE : return "CHECK_GE"; + case assertType::DT_REQUIRE_GE : return "REQUIRE_GE"; + case assertType::DT_WARN_LE : return "WARN_LE"; + case assertType::DT_CHECK_LE : return "CHECK_LE"; + case assertType::DT_REQUIRE_LE : return "REQUIRE_LE"; + + case assertType::DT_WARN_UNARY : return "WARN_UNARY"; + case assertType::DT_CHECK_UNARY : return "CHECK_UNARY"; + case assertType::DT_REQUIRE_UNARY : return "REQUIRE_UNARY"; + case assertType::DT_WARN_UNARY_FALSE : return "WARN_UNARY_FALSE"; + case assertType::DT_CHECK_UNARY_FALSE : return "CHECK_UNARY_FALSE"; + case assertType::DT_REQUIRE_UNARY_FALSE : return "REQUIRE_UNARY_FALSE"; + } + DOCTEST_MSVC_SUPPRESS_WARNING_POP + return ""; +} +// clang-format on + +const char* failureString(assertType::Enum at) { + if(at & assertType::is_warn) //!OCLINT bitwise operator in conditional + return "WARNING: "; + if(at & assertType::is_check) //!OCLINT bitwise operator in conditional + return "ERROR: "; + if(at & assertType::is_require) //!OCLINT bitwise operator in conditional + return "FATAL ERROR: "; + return ""; +} + +DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wnull-dereference") +DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wnull-dereference") +// depending on the current options this will remove the path of filenames +const char* removePathFromFilename(const char* file) { + if(getContextOptions()->no_path_in_filenames) { + auto back = std::strrchr(file, '\\'); + auto forward = std::strrchr(file, '/'); + if(back || forward) { + if(back > forward) + forward = back; + return forward + 1; + } + } + return file; +} +DOCTEST_CLANG_SUPPRESS_WARNING_POP +DOCTEST_GCC_SUPPRESS_WARNING_POP + +DOCTEST_DEFINE_DEFAULTS(TestCaseData); +DOCTEST_DEFINE_COPIES(TestCaseData); + +DOCTEST_DEFINE_DEFAULTS(AssertData); + +DOCTEST_DEFINE_DEFAULTS(MessageData); + +SubcaseSignature::SubcaseSignature(const char* name, const char* file, int line) + : m_name(name) + , m_file(file) + , m_line(line) {} + +DOCTEST_DEFINE_DEFAULTS(SubcaseSignature); +DOCTEST_DEFINE_COPIES(SubcaseSignature); + +bool SubcaseSignature::operator<(const SubcaseSignature& other) const { + if(m_line != other.m_line) + return m_line < other.m_line; + if(std::strcmp(m_file, other.m_file) != 0) + return std::strcmp(m_file, other.m_file) < 0; + return std::strcmp(m_name, other.m_name) < 0; +} + +IContextScope::IContextScope() = default; +IContextScope::~IContextScope() = default; + +DOCTEST_DEFINE_DEFAULTS(ContextOptions); + +#ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +String toString(char* in) { return toString(static_cast(in)); } +String toString(const char* in) { return String("\"") + (in ? in : "{null string}") + "\""; } +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING +String toString(bool in) { return in ? "true" : "false"; } +String toString(float in) { return fpToString(in, 5) + "f"; } +String toString(double in) { return fpToString(in, 10); } +String toString(double long in) { return fpToString(in, 15); } + +#define DOCTEST_TO_STRING_OVERLOAD(type, fmt) \ + String toString(type in) { \ + char buf[64]; \ + std::sprintf(buf, fmt, in); \ + return buf; \ + } + +DOCTEST_TO_STRING_OVERLOAD(char, "%d") +DOCTEST_TO_STRING_OVERLOAD(char signed, "%d") +DOCTEST_TO_STRING_OVERLOAD(char unsigned, "%u") +DOCTEST_TO_STRING_OVERLOAD(int short, "%d") +DOCTEST_TO_STRING_OVERLOAD(int short unsigned, "%u") +DOCTEST_TO_STRING_OVERLOAD(int, "%d") +DOCTEST_TO_STRING_OVERLOAD(unsigned, "%u") +DOCTEST_TO_STRING_OVERLOAD(int long, "%ld") +DOCTEST_TO_STRING_OVERLOAD(int long unsigned, "%lu") +DOCTEST_TO_STRING_OVERLOAD(int long long, "%lld") +DOCTEST_TO_STRING_OVERLOAD(int long long unsigned, "%llu") + +String toString(std::nullptr_t) { return "NULL"; } + +Approx::Approx(double value) + : m_epsilon(static_cast(std::numeric_limits::epsilon()) * 100) + , m_scale(1.0) + , m_value(value) {} + +DOCTEST_DEFINE_COPIES(Approx); + +Approx Approx::operator()(double value) const { + Approx approx(value); + approx.epsilon(m_epsilon); + approx.scale(m_scale); + return approx; +} + +Approx& Approx::epsilon(double newEpsilon) { + m_epsilon = newEpsilon; + return *this; +} +Approx& Approx::scale(double newScale) { + m_scale = newScale; + return *this; +} + +bool operator==(double lhs, const Approx& rhs) { + // Thanks to Richard Harris for his help refining this formula + return std::fabs(lhs - rhs.m_value) < + rhs.m_epsilon * (rhs.m_scale + std::max(std::fabs(lhs), std::fabs(rhs.m_value))); +} +bool operator==(const Approx& lhs, double rhs) { return operator==(rhs, lhs); } +bool operator!=(double lhs, const Approx& rhs) { return !operator==(lhs, rhs); } +bool operator!=(const Approx& lhs, double rhs) { return !operator==(rhs, lhs); } +bool operator<=(double lhs, const Approx& rhs) { return lhs < rhs.m_value || lhs == rhs; } +bool operator<=(const Approx& lhs, double rhs) { return lhs.m_value < rhs || lhs == rhs; } +bool operator>=(double lhs, const Approx& rhs) { return lhs > rhs.m_value || lhs == rhs; } +bool operator>=(const Approx& lhs, double rhs) { return lhs.m_value > rhs || lhs == rhs; } +bool operator<(double lhs, const Approx& rhs) { return lhs < rhs.m_value && lhs != rhs; } +bool operator<(const Approx& lhs, double rhs) { return lhs.m_value < rhs && lhs != rhs; } +bool operator>(double lhs, const Approx& rhs) { return lhs > rhs.m_value && lhs != rhs; } +bool operator>(const Approx& lhs, double rhs) { return lhs.m_value > rhs && lhs != rhs; } + +String toString(const Approx& in) { + return String("Approx( ") + doctest::toString(in.m_value) + " )"; +} +const ContextOptions* getContextOptions() { return DOCTEST_BRANCH_ON_DISABLED(nullptr, g_cs); } + +} // namespace doctest + +#ifdef DOCTEST_CONFIG_DISABLE +namespace doctest { +Context::Context(int, const char* const*) {} +Context::~Context() = default; +void Context::applyCommandLine(int, const char* const*) {} +void Context::addFilter(const char*, const char*) {} +void Context::clearFilters() {} +void Context::setOption(const char*, int) {} +void Context::setOption(const char*, const char*) {} +bool Context::shouldExit() { return false; } +void Context::setAsDefaultForAssertsOutOfTestCases() {} +void Context::setAssertHandler(detail::assert_handler) {} +int Context::run() { return 0; } + +DOCTEST_DEFINE_DEFAULTS(CurrentTestCaseStats); + +DOCTEST_DEFINE_DEFAULTS(TestRunStats); + +IReporter::~IReporter() = default; + +int IReporter::get_num_active_contexts() { return 0; } +const IContextScope* const* IReporter::get_active_contexts() { return nullptr; } +int IReporter::get_num_stringified_contexts() { return 0; } +const String* IReporter::get_stringified_contexts() { return nullptr; } + +int registerReporter(const char*, int, IReporter*) { return 0; } + +} // namespace doctest +#else // DOCTEST_CONFIG_DISABLE + +#if !defined(DOCTEST_CONFIG_COLORS_NONE) +#if !defined(DOCTEST_CONFIG_COLORS_WINDOWS) && !defined(DOCTEST_CONFIG_COLORS_ANSI) +#ifdef DOCTEST_PLATFORM_WINDOWS +#define DOCTEST_CONFIG_COLORS_WINDOWS +#else // linux +#define DOCTEST_CONFIG_COLORS_ANSI +#endif // platform +#endif // DOCTEST_CONFIG_COLORS_WINDOWS && DOCTEST_CONFIG_COLORS_ANSI +#endif // DOCTEST_CONFIG_COLORS_NONE + +#if DOCTEST_MSVC || defined(__MINGW32__) +#if DOCTEST_MSVC +#define DOCTEST_WINDOWS_SAL_IN_OPT _In_opt_ +#else // MSVC +#define DOCTEST_WINDOWS_SAL_IN_OPT +#endif // MSVC +extern "C" __declspec(dllimport) void __stdcall OutputDebugStringA( + DOCTEST_WINDOWS_SAL_IN_OPT const char*); +extern "C" __declspec(dllimport) int __stdcall IsDebuggerPresent(); +#endif // MSVC || __MINGW32__ + +#ifdef DOCTEST_CONFIG_COLORS_ANSI +#include +#endif // DOCTEST_CONFIG_COLORS_ANSI + +#ifdef DOCTEST_PLATFORM_WINDOWS + +// defines for a leaner windows.h +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN +#endif // WIN32_LEAN_AND_MEAN +#ifndef VC_EXTRA_LEAN +#define VC_EXTRA_LEAN +#endif // VC_EXTRA_LEAN +#ifndef NOMINMAX +#define NOMINMAX +#endif // NOMINMAX + +DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_BEGIN + +// not sure what AfxWin.h is for - here I do what Catch does +#ifdef __AFXDLL +#include +#else +#include +#endif +#include + +DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_END + +#else // DOCTEST_PLATFORM_WINDOWS + +#include + +#endif // DOCTEST_PLATFORM_WINDOWS + +namespace doctest_detail_test_suite_ns { +// holds the current test suite +doctest::detail::TestSuite& getCurrentTestSuite() { + static doctest::detail::TestSuite data; + return data; +} +} // namespace doctest_detail_test_suite_ns + +namespace doctest { +namespace { + using namespace detail; + typedef std::map, IReporter*> reporterMap; + reporterMap& getReporters() { + static reporterMap data; + return data; + } +} // namespace +namespace detail { +#define DOCTEST_ITERATE_THROUGH_REPORTERS(function, ...) \ + for(auto& curr_rep : g_cs->reporters_currently_used) \ + curr_rep->function(__VA_ARGS__) + + DOCTEST_DEFINE_DEFAULTS(TestFailureException); + DOCTEST_DEFINE_COPIES(TestFailureException); + bool checkIfShouldThrow(assertType::Enum at) { + if(at & assertType::is_require) //!OCLINT bitwise operator in conditional + return true; + + if((at & assertType::is_check) //!OCLINT bitwise operator in conditional + && getContextOptions()->abort_after > 0 && + (g_cs->numAssertsFailed + g_cs->numAssertsFailedForCurrentTestCase_atomic) >= + getContextOptions()->abort_after) + return true; + + return false; + } + + void throwException() { +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS + throw TestFailureException(); +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + } +} // namespace detail + +namespace { + using namespace detail; + // matching of a string against a wildcard mask (case sensitivity configurable) taken from + // https://www.codeproject.com/Articles/1088/Wildcard-string-compare-globbing + int wildcmp(const char* str, const char* wild, bool caseSensitive) { + const char* cp = nullptr; + const char* mp = nullptr; + + while((*str) && (*wild != '*')) { + if((caseSensitive ? (*wild != *str) : (tolower(*wild) != tolower(*str))) && + (*wild != '?')) { + return 0; + } + wild++; + str++; + } + + while(*str) { + if(*wild == '*') { + if(!*++wild) { + return 1; + } + mp = wild; + cp = str + 1; + } else if((caseSensitive ? (*wild == *str) : (tolower(*wild) == tolower(*str))) || + (*wild == '?')) { + wild++; + str++; + } else { + wild = mp; //!OCLINT parameter reassignment + str = cp++; //!OCLINT parameter reassignment + } + } + + while(*wild == '*') { + wild++; + } + return !*wild; + } + + //// C string hash function (djb2) - taken from http://www.cse.yorku.ca/~oz/hash.html + //unsigned hashStr(unsigned const char* str) { + // unsigned long hash = 5381; + // char c; + // while((c = *str++)) + // hash = ((hash << 5) + hash) + c; // hash * 33 + c + // return hash; + //} + + // checks if the name matches any of the filters (and can be configured what to do when empty) + bool matchesAny(const char* name, const std::vector& filters, bool matchEmpty, + bool caseSensitive) { + if(filters.empty() && matchEmpty) + return true; + for(auto& curr : filters) + if(wildcmp(name, curr.c_str(), caseSensitive)) + return true; + return false; + } + +#ifdef DOCTEST_PLATFORM_WINDOWS + + typedef unsigned long long UInt64; + + UInt64 getCurrentTicks() { + static UInt64 hz = 0, hzo = 0; + if(!hz) { + QueryPerformanceFrequency(reinterpret_cast(&hz)); + QueryPerformanceCounter(reinterpret_cast(&hzo)); + } + UInt64 t; + QueryPerformanceCounter(reinterpret_cast(&t)); + return ((t - hzo) * 1000000) / hz; + } +#else // DOCTEST_PLATFORM_WINDOWS + + typedef uint64_t UInt64; + + UInt64 getCurrentTicks() { + timeval t; + gettimeofday(&t, nullptr); + return static_cast(t.tv_sec) * 1000000 + static_cast(t.tv_usec); + } +#endif // DOCTEST_PLATFORM_WINDOWS + + struct Timer + { + void start() { m_ticks = getCurrentTicks(); } + unsigned int getElapsedMicroseconds() const { + return static_cast(getCurrentTicks() - m_ticks); + } + //unsigned int getElapsedMilliseconds() const { + // return static_cast(getElapsedMicroseconds() / 1000); + //} + double getElapsedSeconds() const { return getElapsedMicroseconds() / 1000000.0; } + + private: + UInt64 m_ticks = 0; + }; + + Timer g_timer; +} // namespace +namespace detail { + + Subcase::Subcase(const char* name, const char* file, int line) + : m_signature(name, file, line) { + ContextState* s = g_cs; + + // if we have already completed it + if(s->subcasesPassed.count(m_signature) != 0) + return; + + // check subcase filters + if(s->subcasesCurrentLevel < s->subcase_filter_levels) { + if(!matchesAny(m_signature.m_name, s->filters[6], true, s->case_sensitive)) + return; + if(matchesAny(m_signature.m_name, s->filters[7], false, s->case_sensitive)) + return; + } + + // if a Subcase on the same level has already been entered + if(s->subcasesEnteredLevels.count(s->subcasesCurrentLevel) != 0) { + s->should_reenter = true; + return; + } + + s->subcasesEnteredLevels.insert(s->subcasesCurrentLevel++); + m_entered = true; + + DOCTEST_ITERATE_THROUGH_REPORTERS(subcase_start, m_signature); + } + + Subcase::~Subcase() { + if(m_entered) { + ContextState* s = g_cs; + + s->subcasesCurrentLevel--; + // only mark the subcase as passed if no subcases have been skipped + if(s->should_reenter == false) + s->subcasesPassed.insert(m_signature); + + DOCTEST_ITERATE_THROUGH_REPORTERS(subcase_end, m_signature); + } + } + + Subcase::operator bool() const { return m_entered; } + + Result::Result(bool passed, const String& decomposition) + : m_passed(passed) + , m_decomp(decomposition) {} + + DOCTEST_DEFINE_DEFAULTS(Result); + DOCTEST_DEFINE_COPIES(Result); + + ExpressionDecomposer::ExpressionDecomposer(assertType::Enum at) + : m_at(at) {} + + DOCTEST_DEFINE_DEFAULTS(ExpressionDecomposer); + + DOCTEST_DEFINE_DEFAULTS(TestSuite); + DOCTEST_DEFINE_COPIES(TestSuite); + + TestSuite& TestSuite::operator*(const char* in) { + m_test_suite = in; + // clear state + m_description = nullptr; + m_skip = false; + m_may_fail = false; + m_should_fail = false; + m_expected_failures = 0; + m_timeout = 0; + return *this; + } + + TestCase::TestCase(funcType test, const char* file, unsigned line, const TestSuite& test_suite, + const char* type, int template_id) { + m_file = file; + m_line = line; + m_name = nullptr; + m_test_suite = test_suite.m_test_suite; + m_description = test_suite.m_description; + m_skip = test_suite.m_skip; + m_may_fail = test_suite.m_may_fail; + m_should_fail = test_suite.m_should_fail; + m_expected_failures = test_suite.m_expected_failures; + m_timeout = test_suite.m_timeout; + + m_test = test; + m_type = type; + m_template_id = template_id; + } + + DOCTEST_DEFINE_DEFAULTS(TestCase); + + TestCase::TestCase(const TestCase& other) + : TestCaseData() { + *this = other; + } + + DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(26434) // hides a non-virtual function + DOCTEST_MSVC_SUPPRESS_WARNING(26437) // Do not slice + TestCase& TestCase::operator=(const TestCase& other) { + static_cast(*this) = static_cast(other); + + m_test = other.m_test; + m_type = other.m_type; + m_template_id = other.m_template_id; + m_full_name = other.m_full_name; + + if(m_template_id != -1) + m_name = m_full_name.c_str(); + return *this; + } + DOCTEST_MSVC_SUPPRESS_WARNING_POP + + TestCase& TestCase::operator*(const char* in) { + m_name = in; + // make a new name with an appended type for templated test case + if(m_template_id != -1) { + m_full_name = String(m_name) + m_type; + // redirect the name to point to the newly constructed full name + m_name = m_full_name.c_str(); + } + return *this; + } + + bool TestCase::operator<(const TestCase& other) const { + if(m_line != other.m_line) + return m_line < other.m_line; + const int file_cmp = std::strcmp(m_file, other.m_file); + if(file_cmp != 0) + return file_cmp < 0; + return m_template_id < other.m_template_id; + } +} // namespace detail +namespace { + using namespace detail; + // for sorting tests by file/line + int fileOrderComparator(const void* a, const void* b) { + auto lhs = *static_cast(a); + auto rhs = *static_cast(b); +#if DOCTEST_MSVC + // this is needed because MSVC gives different case for drive letters + // for __FILE__ when evaluated in a header and a source file + const int res = stricmp(lhs->m_file, rhs->m_file); +#else // MSVC + const int res = std::strcmp(lhs->m_file, rhs->m_file); +#endif // MSVC + if(res != 0) + return res; + return static_cast(lhs->m_line) - static_cast(rhs->m_line); + } + + // for sorting tests by suite/file/line + int suiteOrderComparator(const void* a, const void* b) { + auto lhs = *static_cast(a); + auto rhs = *static_cast(b); + + const int res = std::strcmp(lhs->m_test_suite, rhs->m_test_suite); + if(res != 0) + return res; + return fileOrderComparator(a, b); + } + + // for sorting tests by name/suite/file/line + int nameOrderComparator(const void* a, const void* b) { + auto lhs = *static_cast(a); + auto rhs = *static_cast(b); + + const int res_name = std::strcmp(lhs->m_name, rhs->m_name); + if(res_name != 0) + return res_name; + return suiteOrderComparator(a, b); + } + + // all the registered tests + std::set& getRegisteredTests() { + static std::set data; + return data; + } + +#ifdef DOCTEST_CONFIG_COLORS_WINDOWS + HANDLE g_stdoutHandle; + WORD g_origFgAttrs; + WORD g_origBgAttrs; + bool g_attrsInitted = false; + + int colors_init() { + if(!g_attrsInitted) { + g_stdoutHandle = GetStdHandle(STD_OUTPUT_HANDLE); + g_attrsInitted = true; + CONSOLE_SCREEN_BUFFER_INFO csbiInfo; + GetConsoleScreenBufferInfo(g_stdoutHandle, &csbiInfo); + g_origFgAttrs = csbiInfo.wAttributes & ~(BACKGROUND_GREEN | BACKGROUND_RED | + BACKGROUND_BLUE | BACKGROUND_INTENSITY); + g_origBgAttrs = csbiInfo.wAttributes & ~(FOREGROUND_GREEN | FOREGROUND_RED | + FOREGROUND_BLUE | FOREGROUND_INTENSITY); + } + return 0; + } + + int dumy_init_console_colors = colors_init(); +#endif // DOCTEST_CONFIG_COLORS_WINDOWS + + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations") + void color_to_stream(std::ostream& s, Color::Enum code) { + ((void)s); // for DOCTEST_CONFIG_COLORS_NONE or DOCTEST_CONFIG_COLORS_WINDOWS + ((void)code); // for DOCTEST_CONFIG_COLORS_NONE +#ifdef DOCTEST_CONFIG_COLORS_ANSI + if(g_no_colors || + (isatty(STDOUT_FILENO) == false && getContextOptions()->force_colors == false)) + return; + + auto col = ""; + // clang-format off + switch(code) { //!OCLINT missing break in switch statement / unnecessary default statement in covered switch statement + case Color::Red: col = "[0;31m"; break; + case Color::Green: col = "[0;32m"; break; + case Color::Blue: col = "[0;34m"; break; + case Color::Cyan: col = "[0;36m"; break; + case Color::Yellow: col = "[0;33m"; break; + case Color::Grey: col = "[1;30m"; break; + case Color::LightGrey: col = "[0;37m"; break; + case Color::BrightRed: col = "[1;31m"; break; + case Color::BrightGreen: col = "[1;32m"; break; + case Color::BrightWhite: col = "[1;37m"; break; + case Color::Bright: // invalid + case Color::None: + case Color::White: + default: col = "[0m"; + } + // clang-format on + s << "\033" << col; +#endif // DOCTEST_CONFIG_COLORS_ANSI + +#ifdef DOCTEST_CONFIG_COLORS_WINDOWS + if(g_no_colors || + (isatty(fileno(stdout)) == false && getContextOptions()->force_colors == false)) + return; + +#define DOCTEST_SET_ATTR(x) SetConsoleTextAttribute(g_stdoutHandle, x | g_origBgAttrs) + + // clang-format off + switch (code) { + case Color::White: DOCTEST_SET_ATTR(FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE); break; + case Color::Red: DOCTEST_SET_ATTR(FOREGROUND_RED); break; + case Color::Green: DOCTEST_SET_ATTR(FOREGROUND_GREEN); break; + case Color::Blue: DOCTEST_SET_ATTR(FOREGROUND_BLUE); break; + case Color::Cyan: DOCTEST_SET_ATTR(FOREGROUND_BLUE | FOREGROUND_GREEN); break; + case Color::Yellow: DOCTEST_SET_ATTR(FOREGROUND_RED | FOREGROUND_GREEN); break; + case Color::Grey: DOCTEST_SET_ATTR(0); break; + case Color::LightGrey: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY); break; + case Color::BrightRed: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_RED); break; + case Color::BrightGreen: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_GREEN); break; + case Color::BrightWhite: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE); break; + case Color::None: + case Color::Bright: // invalid + default: DOCTEST_SET_ATTR(g_origFgAttrs); + } + // clang-format on +#endif // DOCTEST_CONFIG_COLORS_WINDOWS + } + DOCTEST_CLANG_SUPPRESS_WARNING_POP + + std::vector& getExceptionTranslators() { + static std::vector data; + return data; + } + + String translateActiveException() { +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS + String res; + auto& translators = getExceptionTranslators(); + for(auto& curr : translators) + if(curr->translate(res)) + return res; + // clang-format off + try { + throw; + } catch(std::exception& ex) { + return ex.what(); + } catch(std::string& msg) { + return msg.c_str(); + } catch(const char* msg) { + return msg; + } catch(...) { + return "unknown exception"; + } +// clang-format on +#else // DOCTEST_CONFIG_NO_EXCEPTIONS + return ""; +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + } +} // namespace + +namespace detail { + // used by the macros for registering tests + int regTest(const TestCase& tc) { + getRegisteredTests().insert(tc); + return 0; + } + + // sets the current test suite + int setTestSuite(const TestSuite& ts) { + doctest_detail_test_suite_ns::getCurrentTestSuite() = ts; + return 0; + } + +#ifdef DOCTEST_PLATFORM_MAC + // The following function is taken directly from the following technical note: + // http://developer.apple.com/library/mac/#qa/qa2004/qa1361.html + // Returns true if the current process is being debugged (either + // running under the debugger or has a debugger attached post facto). + bool isDebuggerActive() { + int mib[4]; + kinfo_proc info; + size_t size; + // Initialize the flags so that, if sysctl fails for some bizarre + // reason, we get a predictable result. + info.kp_proc.p_flag = 0; + // Initialize mib, which tells sysctl the info we want, in this case + // we're looking for information about a specific process ID. + mib[0] = CTL_KERN; + mib[1] = KERN_PROC; + mib[2] = KERN_PROC_PID; + mib[3] = getpid(); + // Call sysctl. + size = sizeof(info); + if(sysctl(mib, DOCTEST_COUNTOF(mib), &info, &size, 0, 0) != 0) { + fprintf(stderr, "\n** Call to sysctl failed - unable to determine if debugger is " + "active **\n\n"); + return false; + } + // We're being debugged if the P_TRACED flag is set. + return ((info.kp_proc.p_flag & P_TRACED) != 0); + } +#elif DOCTEST_MSVC || defined(__MINGW32__) + bool isDebuggerActive() { return ::IsDebuggerPresent() != 0; } +#else + bool isDebuggerActive() { return false; } +#endif // Platform + + void registerExceptionTranslatorImpl(const IExceptionTranslator* et) { + if(std::find(getExceptionTranslators().begin(), getExceptionTranslators().end(), et) == + getExceptionTranslators().end()) + getExceptionTranslators().push_back(et); + } + + void writeStringToStream(std::ostream* s, const String& str) { *s << str; } + +#ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + void toStream(std::ostream* s, char* in) { *s << in; } + void toStream(std::ostream* s, const char* in) { *s << in; } +#endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING + void toStream(std::ostream* s, bool in) { *s << std::boolalpha << in << std::noboolalpha; } + void toStream(std::ostream* s, float in) { *s << in; } + void toStream(std::ostream* s, double in) { *s << in; } + void toStream(std::ostream* s, double long in) { *s << in; } + + void toStream(std::ostream* s, char in) { *s << in; } + void toStream(std::ostream* s, char signed in) { *s << in; } + void toStream(std::ostream* s, char unsigned in) { *s << in; } + void toStream(std::ostream* s, int short in) { *s << in; } + void toStream(std::ostream* s, int short unsigned in) { *s << in; } + void toStream(std::ostream* s, int in) { *s << in; } + void toStream(std::ostream* s, int unsigned in) { *s << in; } + void toStream(std::ostream* s, int long in) { *s << in; } + void toStream(std::ostream* s, int long unsigned in) { *s << in; } + void toStream(std::ostream* s, int long long in) { *s << in; } + void toStream(std::ostream* s, int long long unsigned in) { *s << in; } + + DOCTEST_THREAD_LOCAL std::vector g_infoContexts; // for logging with INFO() + + ContextBuilder::ICapture::ICapture() = default; + ContextBuilder::ICapture::~ICapture() = default; + + ContextBuilder::Chunk::Chunk() = default; + ContextBuilder::Chunk::~Chunk() = default; + + ContextBuilder::Node::Node() = default; + ContextBuilder::Node::~Node() = default; + + // steal the contents of the other - acting as a move constructor... + ContextBuilder::ContextBuilder(ContextBuilder& other) + : numCaptures(other.numCaptures) + , head(other.head) + , tail(other.tail) { + other.numCaptures = 0; + other.head = nullptr; + other.tail = nullptr; + memcpy(stackChunks, other.stackChunks, + unsigned(int(sizeof(Chunk)) * DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK)); + } + + DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wcast-align") + void ContextBuilder::stringify(std::ostream* s) const { + int curr = 0; + // iterate over small buffer + while(curr < numCaptures && curr < DOCTEST_CONFIG_NUM_CAPTURES_ON_STACK) + reinterpret_cast(stackChunks[curr++].buf)->toStream(s); + // iterate over list + auto curr_elem = head; + while(curr < numCaptures) { + reinterpret_cast(curr_elem->chunk.buf)->toStream(s); + curr_elem = curr_elem->next; + ++curr; + } + } + DOCTEST_GCC_SUPPRESS_WARNING_POP + + ContextBuilder::ContextBuilder() = default; + + ContextBuilder::~ContextBuilder() { + // free the linked list - the ones on the stack are left as-is + // no destructors are called at all - there is no need + while(head) { + auto next = head->next; + delete head; + head = next; + } + } + + ContextScope::ContextScope(ContextBuilder& temp) + : contextBuilder(temp) { + g_infoContexts.push_back(this); + } + + DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4996) // std::uncaught_exception is deprecated in C++17 + DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations") + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations") + ContextScope::~ContextScope() { + if(std::uncaught_exception()) { + std::ostringstream s; + this->stringify(&s); + g_cs->stringifiedContexts.push_back(s.str().c_str()); + } + g_infoContexts.pop_back(); + } + DOCTEST_CLANG_SUPPRESS_WARNING_POP + DOCTEST_GCC_SUPPRESS_WARNING_POP + DOCTEST_MSVC_SUPPRESS_WARNING_POP + + void ContextScope::stringify(std::ostream* s) const { contextBuilder.stringify(s); } +} // namespace detail +namespace { + using namespace detail; +#if !defined(DOCTEST_CONFIG_POSIX_SIGNALS) && !defined(DOCTEST_CONFIG_WINDOWS_SEH) + struct FatalConditionHandler + { + void reset() {} + }; +#else // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH + + void reportFatal(const std::string&); + +#ifdef DOCTEST_PLATFORM_WINDOWS + + struct SignalDefs + { + DWORD id; + const char* name; + }; + // There is no 1-1 mapping between signals and windows exceptions. + // Windows can easily distinguish between SO and SigSegV, + // but SigInt, SigTerm, etc are handled differently. + SignalDefs signalDefs[] = { + {EXCEPTION_ILLEGAL_INSTRUCTION, "SIGILL - Illegal instruction signal"}, + {EXCEPTION_STACK_OVERFLOW, "SIGSEGV - Stack overflow"}, + {EXCEPTION_ACCESS_VIOLATION, "SIGSEGV - Segmentation violation signal"}, + {EXCEPTION_INT_DIVIDE_BY_ZERO, "Divide by zero error"}, + }; + + struct FatalConditionHandler + { + static LONG CALLBACK handleVectoredException(PEXCEPTION_POINTERS ExceptionInfo) { + for(size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) { + if(ExceptionInfo->ExceptionRecord->ExceptionCode == signalDefs[i].id) { + reportFatal(signalDefs[i].name); + } + } + // If its not an exception we care about, pass it along. + // This stops us from eating debugger breaks etc. + return EXCEPTION_CONTINUE_SEARCH; + } + + FatalConditionHandler() { + isSet = true; + // 32k seems enough for doctest to handle stack overflow, + // but the value was found experimentally, so there is no strong guarantee + guaranteeSize = 32 * 1024; + exceptionHandlerHandle = nullptr; + // Register as first handler in current chain + exceptionHandlerHandle = AddVectoredExceptionHandler(1, handleVectoredException); + // Pass in guarantee size to be filled + SetThreadStackGuarantee(&guaranteeSize); + } + + static void reset() { + if(isSet) { + // Unregister handler and restore the old guarantee + RemoveVectoredExceptionHandler(exceptionHandlerHandle); + SetThreadStackGuarantee(&guaranteeSize); + exceptionHandlerHandle = nullptr; + isSet = false; + } + } + + ~FatalConditionHandler() { reset(); } + + private: + static bool isSet; + static ULONG guaranteeSize; + static PVOID exceptionHandlerHandle; + }; + + bool FatalConditionHandler::isSet = false; + ULONG FatalConditionHandler::guaranteeSize = 0; + PVOID FatalConditionHandler::exceptionHandlerHandle = nullptr; + +#else // DOCTEST_PLATFORM_WINDOWS + + struct SignalDefs + { + int id; + const char* name; + }; + SignalDefs signalDefs[] = {{SIGINT, "SIGINT - Terminal interrupt signal"}, + {SIGILL, "SIGILL - Illegal instruction signal"}, + {SIGFPE, "SIGFPE - Floating point error signal"}, + {SIGSEGV, "SIGSEGV - Segmentation violation signal"}, + {SIGTERM, "SIGTERM - Termination request signal"}, + {SIGABRT, "SIGABRT - Abort (abnormal termination) signal"}}; + + struct FatalConditionHandler + { + static bool isSet; + static struct sigaction oldSigActions[DOCTEST_COUNTOF(signalDefs)]; + static stack_t oldSigStack; + static char altStackMem[4 * SIGSTKSZ]; + + static void handleSignal(int sig) { + const char* name = ""; + for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) { + SignalDefs& def = signalDefs[i]; + if(sig == def.id) { + name = def.name; + break; + } + } + reset(); + reportFatal(name); + raise(sig); + } + + FatalConditionHandler() { + isSet = true; + stack_t sigStack; + sigStack.ss_sp = altStackMem; + sigStack.ss_size = sizeof(altStackMem); + sigStack.ss_flags = 0; + sigaltstack(&sigStack, &oldSigStack); + struct sigaction sa = {}; + sa.sa_handler = handleSignal; // NOLINT + sa.sa_flags = SA_ONSTACK; + for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) { + sigaction(signalDefs[i].id, &sa, &oldSigActions[i]); + } + } + + ~FatalConditionHandler() { reset(); } + static void reset() { + if(isSet) { + // Set signals back to previous values -- hopefully nobody overwrote them in the meantime + for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) { + sigaction(signalDefs[i].id, &oldSigActions[i], nullptr); + } + // Return the old stack + sigaltstack(&oldSigStack, nullptr); + isSet = false; + } + } + }; + + bool FatalConditionHandler::isSet = false; + struct sigaction FatalConditionHandler::oldSigActions[DOCTEST_COUNTOF(signalDefs)] = {}; + stack_t FatalConditionHandler::oldSigStack = {}; + char FatalConditionHandler::altStackMem[] = {}; + +#endif // DOCTEST_PLATFORM_WINDOWS +#endif // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH + +} // namespace + +namespace { + using namespace detail; + +#ifdef DOCTEST_PLATFORM_WINDOWS +#define DOCTEST_OUTPUT_DEBUG_STRING(text) ::OutputDebugStringA(text) +#else + // TODO: integration with XCode and other IDEs +#define DOCTEST_OUTPUT_DEBUG_STRING(text) +#endif // Platform + + void addAssert(assertType::Enum at) { + if((at & assertType::is_warn) == 0) //!OCLINT bitwise operator in conditional + g_cs->numAssertsForCurrentTestCase_atomic++; + } + + void addFailedAssert(assertType::Enum at) { + if((at & assertType::is_warn) == 0) //!OCLINT bitwise operator in conditional + g_cs->numAssertsFailedForCurrentTestCase_atomic++; + } + +#if defined(DOCTEST_CONFIG_POSIX_SIGNALS) || defined(DOCTEST_CONFIG_WINDOWS_SEH) + void reportFatal(const std::string& message) { + g_cs->seconds_so_far += g_timer.getElapsedSeconds(); + g_cs->failure_flags |= TestCaseFailureReason::Crash; + g_cs->error_string = message.c_str(); + g_cs->should_reenter = false; + + // TODO: end all currently opened subcases...? + + DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_end, *g_cs); + + g_cs->numTestCasesFailed++; + + DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_end, *g_cs); + } +#endif // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH +} // namespace +namespace detail { + + ResultBuilder::ResultBuilder(assertType::Enum at, const char* file, int line, const char* expr, + const char* exception_type) { + m_test_case = g_cs->currentTest; + m_at = at; + m_file = file; + m_line = line; + m_expr = expr; + m_failed = true; + m_threw = false; + m_threw_as = false; + m_exception_type = exception_type; +#if DOCTEST_MSVC + if(m_expr[0] == ' ') // this happens when variadic macros are disabled under MSVC + ++m_expr; +#endif // MSVC + } + + DOCTEST_DEFINE_DEFAULTS(ResultBuilder); + + void ResultBuilder::setResult(const Result& res) { + m_decomp = res.m_decomp; + m_failed = !res.m_passed; + } + + void ResultBuilder::translateException() { + m_threw = true; + m_exception = translateActiveException(); + } + + bool ResultBuilder::log() { + if(m_at & assertType::is_throws) { //!OCLINT bitwise operator in conditional + m_failed = !m_threw; + } else if(m_at & assertType::is_throws_as) { //!OCLINT bitwise operator in conditional + m_failed = !m_threw_as; + } else if(m_at & assertType::is_throws_with) { //!OCLINT bitwise operator in conditional + m_failed = m_exception != m_exception_type; + } else if(m_at & assertType::is_nothrow) { //!OCLINT bitwise operator in conditional + m_failed = m_threw; + } + + if(m_exception.size()) + m_exception = String("\"") + m_exception + "\""; + + if(is_running_in_test) { + addAssert(m_at); + DOCTEST_ITERATE_THROUGH_REPORTERS(log_assert, *this); + + if(m_failed) + addFailedAssert(m_at); + } else if(m_failed) { + failed_out_of_a_testing_context(*this); + } + + return m_failed && isDebuggerActive() && + !getContextOptions()->no_breaks; // break into debugger + } + + void ResultBuilder::react() const { + if(m_failed && checkIfShouldThrow(m_at)) + throwException(); + } + + void failed_out_of_a_testing_context(const AssertData& ad) { + if(g_cs->ah) + g_cs->ah(ad); + else + std::abort(); + } + + void decomp_assert(assertType::Enum at, const char* file, int line, const char* expr, + Result result) { + bool failed = !result.m_passed; + + // ################################################################################### + // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT + // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED + // ################################################################################### + DOCTEST_ASSERT_OUT_OF_TESTS(result.m_decomp); + DOCTEST_ASSERT_IN_TESTS(result.m_decomp); + } + + MessageBuilder::MessageBuilder(const char* file, int line, assertType::Enum severity) { + m_stream = getTlsOss(); + m_file = file; + m_line = line; + m_severity = severity; + } + + IExceptionTranslator::IExceptionTranslator() = default; + IExceptionTranslator::~IExceptionTranslator() = default; + + bool MessageBuilder::log() { + m_string = getTlsOssResult(); + DOCTEST_ITERATE_THROUGH_REPORTERS(log_message, *this); + + const bool isWarn = m_severity & assertType::is_warn; + + // warn is just a message in this context so we dont treat it as an assert + if(!isWarn) { + addAssert(m_severity); + addFailedAssert(m_severity); + } + + return isDebuggerActive() && !getContextOptions()->no_breaks && !isWarn; // break + } + + void MessageBuilder::react() { + if(m_severity & assertType::is_require) //!OCLINT bitwise operator in conditional + throwException(); + } + + MessageBuilder::~MessageBuilder() = default; +} // namespace detail +namespace { + std::mutex g_mutex; + + using namespace detail; + struct ConsoleReporter : public IReporter + { + std::ostream& s; + bool hasLoggedCurrentTestStart; + std::vector subcasesStack; + + // caching pointers to objects of these types - safe to do + const ContextOptions* opt; + const TestCaseData* tc; + + ConsoleReporter(std::ostream& in) + : s(in) {} + + // ========================================================================================= + // WHAT FOLLOWS ARE HELPERS USED BY THE OVERRIDES OF THE VIRTUAL METHODS OF THE INTERFACE + // ========================================================================================= + + void separator_to_stream() { + s << Color::Yellow + << "=============================================================================" + "==" + "\n"; + } + + void file_line_to_stream(const char* file, int line, const char* tail = "") { + s << Color::LightGrey << removePathFromFilename(file) + << (opt->gnu_file_line ? ":" : "(") + << (opt->no_line_numbers ? 0 : line) // 0 or the real num depending on the option + << (opt->gnu_file_line ? ":" : "):") << tail; + } + + const char* getSuccessOrFailString(bool success, assertType::Enum at, + const char* success_str) { + if(success) + return success_str; + return failureString(at); + } + + Color::Enum getSuccessOrFailColor(bool success, assertType::Enum at) { + return success ? Color::BrightGreen : + (at & assertType::is_warn) ? Color::Yellow : Color::Red; + } + + void successOrFailColoredStringToStream(bool success, assertType::Enum at, + const char* success_str = "SUCCESS: ") { + s << getSuccessOrFailColor(success, at) + << getSuccessOrFailString(success, at, success_str); + } + + void log_contexts() { + int num_contexts = get_num_active_contexts(); + if(num_contexts) { + auto contexts = get_active_contexts(); + + s << Color::None << " logged: "; + for(int i = 0; i < num_contexts; ++i) { + s << (i == 0 ? "" : " "); + contexts[i]->stringify(&s); + s << "\n"; + } + } + + s << "\n"; + } + + void logTestStart() { + if(hasLoggedCurrentTestStart) + return; + + separator_to_stream(); + file_line_to_stream(tc->m_file, tc->m_line, "\n"); + if(tc->m_description) + s << Color::Yellow << "DESCRIPTION: " << Color::None << tc->m_description << "\n"; + if(tc->m_test_suite && tc->m_test_suite[0] != '\0') + s << Color::Yellow << "TEST SUITE: " << Color::None << tc->m_test_suite << "\n"; + if(strncmp(tc->m_name, " Scenario:", 11) != 0) + s << Color::None << "TEST CASE: "; + s << Color::None << tc->m_name << "\n"; + + for(auto& curr : subcasesStack) + if(curr.m_name[0] != '\0') + s << " " << curr.m_name << "\n"; + + s << "\n"; + + hasLoggedCurrentTestStart = true; + } + + // ========================================================================================= + // WHAT FOLLOWS ARE OVERRIDES OF THE VIRTUAL METHODS OF THE REPORTER INTERFACE + // ========================================================================================= + + void test_run_start(const ContextOptions& o) override { opt = &o; } + + void test_run_end(const TestRunStats& p) override { + separator_to_stream(); + + const bool anythingFailed = p.numTestCasesFailed > 0 || p.numAssertsFailed > 0; + s << Color::Cyan << "[doctest] " << Color::None << "test cases: " << std::setw(6) + << p.numTestCasesPassingFilters << " | " + << ((p.numTestCasesPassingFilters == 0 || anythingFailed) ? Color::None : + Color::Green) + << std::setw(6) << p.numTestCasesPassingFilters - p.numTestCasesFailed << " passed" + << Color::None << " | " << (p.numTestCasesFailed > 0 ? Color::Red : Color::None) + << std::setw(6) << p.numTestCasesFailed << " failed" << Color::None << " | "; + if(opt->no_skipped_summary == false) { + const int numSkipped = p.numTestCases - p.numTestCasesPassingFilters; + s << (numSkipped == 0 ? Color::None : Color::Yellow) << std::setw(6) << numSkipped + << " skipped" << Color::None; + } + s << "\n"; + s << Color::Cyan << "[doctest] " << Color::None << "assertions: " << std::setw(6) + << p.numAsserts << " | " + << ((p.numAsserts == 0 || anythingFailed) ? Color::None : Color::Green) + << std::setw(6) << (p.numAsserts - p.numAssertsFailed) << " passed" << Color::None + << " | " << (p.numAssertsFailed > 0 ? Color::Red : Color::None) << std::setw(6) + << p.numAssertsFailed << " failed" << Color::None << " |\n"; + s << Color::Cyan << "[doctest] " << Color::None + << "Status: " << (p.numTestCasesFailed > 0 ? Color::Red : Color::Green) + << ((p.numTestCasesFailed > 0) ? "FAILURE!\n" : "SUCCESS!\n") << Color::None; + } + + void test_case_start(const TestCaseData& in) override { + hasLoggedCurrentTestStart = false; + tc = ∈ + } + + void test_case_end(const CurrentTestCaseStats& st) override { + // log the preamble of the test case only if there is something + // else to print - something other than that an assert has failed + if(opt->duration || + (st.failure_flags && st.failure_flags != TestCaseFailureReason::AssertFailure)) + logTestStart(); + + // report test case exceptions and crashes + bool crashed = st.failure_flags & TestCaseFailureReason::Crash; + if(crashed || (st.failure_flags & TestCaseFailureReason::Exception)) { + file_line_to_stream(tc->m_file, tc->m_line, " "); + successOrFailColoredStringToStream(false, crashed ? assertType::is_require : + assertType::is_check); + s << Color::Red << (crashed ? "test case CRASHED: " : "test case THREW exception: ") + << Color::Cyan << st.error_string << "\n"; + + int num_stringified_contexts = get_num_stringified_contexts(); + if(num_stringified_contexts) { + auto stringified_contexts = get_stringified_contexts(); + s << Color::None << " logged: "; + for(int i = num_stringified_contexts - 1; i >= 0; --i) { + s << (i == num_stringified_contexts - 1 ? "" : " ") + << stringified_contexts[i] << "\n"; + } + } + s << "\n"; + } + + // means the test case will be re-entered because there are untraversed (but discovered) subcases + if(st.should_reenter) + return; + + if(opt->duration) + s << Color::None << std::setprecision(6) << std::fixed << st.seconds_so_far + << " s: " << tc->m_name << "\n"; + + if(st.failure_flags & TestCaseFailureReason::Timeout) + s << Color::Red << "Test case exceeded time limit of " << std::setprecision(6) + << std::fixed << tc->m_timeout << "!\n"; + + if(st.failure_flags & TestCaseFailureReason::ShouldHaveFailedButDidnt) { + s << Color::Red << "Should have failed but didn't! Marking it as failed!\n"; + } else if(st.failure_flags & TestCaseFailureReason::ShouldHaveFailedAndDid) { + s << Color::Yellow << "Failed as expected so marking it as not failed\n"; + } else if(st.failure_flags & TestCaseFailureReason::CouldHaveFailedAndDid) { + s << Color::Yellow << "Allowed to fail so marking it as not failed\n"; + } else if(st.failure_flags & TestCaseFailureReason::DidntFailExactlyNumTimes) { + s << Color::Red << "Didn't fail exactly " << tc->m_expected_failures + << " times so marking it as failed!\n"; + } else if(st.failure_flags & TestCaseFailureReason::FailedExactlyNumTimes) { + s << Color::Yellow << "Failed exactly " << tc->m_expected_failures + << " times as expected so marking it as not failed!\n"; + } + if(st.failure_flags & TestCaseFailureReason::TooManyFailedAsserts) { + s << Color::Red << "Aborting - too many failed asserts!\n"; + } + s << Color::None; + } + + void subcase_start(const SubcaseSignature& subc) override { + subcasesStack.push_back(subc); + hasLoggedCurrentTestStart = false; + } + + void subcase_end(const SubcaseSignature& /*subc*/) override { + subcasesStack.pop_back(); + hasLoggedCurrentTestStart = false; + } + + void log_assert(const AssertData& rb) override { + if(!rb.m_failed && !opt->success) + return; + + std::lock_guard lock(g_mutex); + + logTestStart(); + + file_line_to_stream(rb.m_file, rb.m_line, " "); + successOrFailColoredStringToStream(!rb.m_failed, rb.m_at); + if((rb.m_at & (assertType::is_throws_as | assertType::is_throws_with)) == + 0) //!OCLINT bitwise operator in conditional + s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << " ) " + << Color::None; + + if(rb.m_at & assertType::is_throws) { //!OCLINT bitwise operator in conditional + s << (rb.m_threw ? "threw as expected!" : "did NOT throw at all!") << "\n"; + } else if(rb.m_at & + assertType::is_throws_as) { //!OCLINT bitwise operator in conditional + s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << ", " + << rb.m_exception_type << " ) " << Color::None + << (rb.m_threw ? (rb.m_threw_as ? "threw as expected!" : + "threw a DIFFERENT exception: ") : + "did NOT throw at all!") + << Color::Cyan << rb.m_exception << "\n"; + } else if(rb.m_at & + assertType::is_throws_with) { //!OCLINT bitwise operator in conditional + s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << ", \"" + << rb.m_exception_type << "\" ) " << Color::None + << (rb.m_threw ? (!rb.m_failed ? "threw as expected!" : + "threw a DIFFERENT exception: ") : + "did NOT throw at all!") + << Color::Cyan << rb.m_exception << "\n"; + } else if(rb.m_at & assertType::is_nothrow) { //!OCLINT bitwise operator in conditional + s << (rb.m_threw ? "THREW exception: " : "didn't throw!") << Color::Cyan + << rb.m_exception << "\n"; + } else { + s << (rb.m_threw ? "THREW exception: " : + (!rb.m_failed ? "is correct!\n" : "is NOT correct!\n")); + if(rb.m_threw) + s << rb.m_exception << "\n"; + else + s << " values: " << assertString(rb.m_at) << "( " << rb.m_decomp << " )\n"; + } + + log_contexts(); + } + + void log_message(const MessageData& mb) override { + std::lock_guard lock(g_mutex); + + logTestStart(); + + file_line_to_stream(mb.m_file, mb.m_line, " "); + s << getSuccessOrFailColor(false, mb.m_severity) + << getSuccessOrFailString(mb.m_severity & assertType::is_warn, mb.m_severity, + "MESSAGE: "); + s << Color::None << mb.m_string << "\n"; + log_contexts(); + } + + void test_case_skipped(const TestCaseData&) override {} + }; + + struct Whitespace + { + int nrSpaces; + explicit Whitespace(int nr) + : nrSpaces(nr) {} + }; + + std::ostream& operator<<(std::ostream& out, const Whitespace& ws) { + if(ws.nrSpaces != 0) + out << std::setw(ws.nrSpaces) << ' '; + return out; + } + + // extension of the console reporter - with a bunch of helpers for the stdout stream redirection + struct ConsoleReporterWithHelpers : public ConsoleReporter + { + ConsoleReporterWithHelpers(std::ostream& in) + : ConsoleReporter(in) {} + + void printVersion() { + if(getContextOptions()->no_version == false) + s << Color::Cyan << "[doctest] " << Color::None << "doctest version is \"" + << DOCTEST_VERSION_STR << "\"\n"; + } + + void printIntro() { + printVersion(); + s << Color::Cyan << "[doctest] " << Color::None + << "run with \"--" DOCTEST_OPTIONS_PREFIX_DISPLAY "help\" for options\n"; + } + + void printHelp() { + int sizePrefixDisplay = static_cast(strlen(DOCTEST_OPTIONS_PREFIX_DISPLAY)); + printVersion(); + // clang-format off + s << Color::Cyan << "[doctest]\n" << Color::None; + s << Color::Cyan << "[doctest] " << Color::None; + s << "boolean values: \"1/on/yes/true\" or \"0/off/no/false\"\n"; + s << Color::Cyan << "[doctest] " << Color::None; + s << "filter values: \"str1,str2,str3\" (comma separated strings)\n"; + s << Color::Cyan << "[doctest]\n" << Color::None; + s << Color::Cyan << "[doctest] " << Color::None; + s << "filters use wildcards for matching strings\n"; + s << Color::Cyan << "[doctest] " << Color::None; + s << "something passes a filter if any of the strings in a filter matches\n"; +#ifndef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS + s << Color::Cyan << "[doctest]\n" << Color::None; + s << Color::Cyan << "[doctest] " << Color::None; + s << "ALL FLAGS, OPTIONS AND FILTERS ALSO AVAILABLE WITH A \"" DOCTEST_CONFIG_OPTIONS_PREFIX "\" PREFIX!!!\n"; +#endif + s << Color::Cyan << "[doctest]\n" << Color::None; + s << Color::Cyan << "[doctest] " << Color::None; + s << "Query flags - the program quits after them. Available:\n\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "?, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "help, -" DOCTEST_OPTIONS_PREFIX_DISPLAY "h " + << Whitespace(sizePrefixDisplay*0) << "prints this message\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "v, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "version " + << Whitespace(sizePrefixDisplay*1) << "prints the version\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "c, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "count " + << Whitespace(sizePrefixDisplay*1) << "prints the number of matching tests\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ltc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-test-cases " + << Whitespace(sizePrefixDisplay*1) << "lists all matching tests by name\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "lts, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-test-suites " + << Whitespace(sizePrefixDisplay*1) << "lists all matching test suites\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "lr, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-reporters " + << Whitespace(sizePrefixDisplay*1) << "lists all registered reporters\n\n"; + // ================================================================================== << 79 + s << Color::Cyan << "[doctest] " << Color::None; + s << "The available / options/filters are:\n\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-case= " + << Whitespace(sizePrefixDisplay*1) << "filters tests by their name\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tce, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-case-exclude= " + << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their name\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sf, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "source-file= " + << Whitespace(sizePrefixDisplay*1) << "filters tests by their file\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sfe, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "source-file-exclude= " + << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their file\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ts, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-suite= " + << Whitespace(sizePrefixDisplay*1) << "filters tests by their test suite\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tse, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-suite-exclude= " + << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their test suite\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase= " + << Whitespace(sizePrefixDisplay*1) << "filters subcases by their name\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sce, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase-exclude= " + << Whitespace(sizePrefixDisplay*1) << "filters OUT subcases by their name\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "r, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "reporters= " + << Whitespace(sizePrefixDisplay*1) << "reporters to use (console is default)\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ob, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "order-by= " + << Whitespace(sizePrefixDisplay*1) << "how the tests should be ordered\n"; + s << Whitespace(sizePrefixDisplay*3) << " - by [file/suite/name/rand]\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "rs, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "rand-seed= " + << Whitespace(sizePrefixDisplay*1) << "seed for random ordering\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "f, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "first= " + << Whitespace(sizePrefixDisplay*1) << "the first test passing the filters to\n"; + s << Whitespace(sizePrefixDisplay*3) << " execute - for range-based execution\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "l, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "last= " + << Whitespace(sizePrefixDisplay*1) << "the last test passing the filters to\n"; + s << Whitespace(sizePrefixDisplay*3) << " execute - for range-based execution\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "aa, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "abort-after= " + << Whitespace(sizePrefixDisplay*1) << "stop after failed assertions\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "scfl,--" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase-filter-levels= " + << Whitespace(sizePrefixDisplay*1) << "apply filters for the first levels\n"; + s << Color::Cyan << "\n[doctest] " << Color::None; + s << "Bool options - can be used like flags and true is assumed. Available:\n\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "s, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "success= " + << Whitespace(sizePrefixDisplay*1) << "include successful assertions in output\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "cs, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "case-sensitive= " + << Whitespace(sizePrefixDisplay*1) << "filters being treated as case sensitive\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "e, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "exit= " + << Whitespace(sizePrefixDisplay*1) << "exits after the tests finish\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "d, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "duration= " + << Whitespace(sizePrefixDisplay*1) << "prints the time duration of each test\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nt, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-throw= " + << Whitespace(sizePrefixDisplay*1) << "skips exceptions-related assert checks\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ne, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-exitcode= " + << Whitespace(sizePrefixDisplay*1) << "returns (or exits) always with success\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nr, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-run= " + << Whitespace(sizePrefixDisplay*1) << "skips all runtime doctest operations\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nv, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-version= " + << Whitespace(sizePrefixDisplay*1) << "omit the framework version in the output\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-colors= " + << Whitespace(sizePrefixDisplay*1) << "disables colors in output\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "fc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "force-colors= " + << Whitespace(sizePrefixDisplay*1) << "use colors even when not in a tty\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nb, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-breaks= " + << Whitespace(sizePrefixDisplay*1) << "disables breakpoints in debuggers\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ns, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-skip= " + << Whitespace(sizePrefixDisplay*1) << "don't skip test cases marked as skip\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "gfl, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "gnu-file-line= " + << Whitespace(sizePrefixDisplay*1) << ":n: vs (n): for line numbers in output\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "npf, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-path-filenames= " + << Whitespace(sizePrefixDisplay*1) << "only filenames and no paths in output\n"; + s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nln, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-line-numbers= " + << Whitespace(sizePrefixDisplay*1) << "0 instead of real line numbers in output\n"; + // ================================================================================== << 79 + // clang-format on + + s << Color::Cyan << "\n[doctest] " << Color::None; + s << "for more information visit the project documentation\n\n"; + } + + void printRegisteredReporters() { + printVersion(); + s << Color::Cyan << "[doctest] " << Color::None << "listing all registered reporters\n"; + for(auto& curr : getReporters()) + s << "priority: " << std::setw(5) << curr.first.first + << " name: " << curr.first.second << "\n"; + } + + void output_query_results() { + separator_to_stream(); + if(getContextOptions()->count || getContextOptions()->list_test_cases) { + s << Color::Cyan << "[doctest] " << Color::None + << "unskipped test cases passing the current filters: " + << g_cs->numTestCasesPassingFilters << "\n"; + } else if(getContextOptions()->list_test_suites) { + s << Color::Cyan << "[doctest] " << Color::None + << "unskipped test cases passing the current filters: " + << g_cs->numTestCasesPassingFilters << "\n"; + s << Color::Cyan << "[doctest] " << Color::None + << "test suites with unskipped test cases passing the current filters: " + << g_cs->numTestSuitesPassingFilters << "\n"; + } + } + + void output_query_preamble_test_cases() { + s << Color::Cyan << "[doctest] " << Color::None << "listing all test case names\n"; + separator_to_stream(); + } + + void output_query_preamble_test_suites() { + s << Color::Cyan << "[doctest] " << Color::None << "listing all test suites\n"; + separator_to_stream(); + } + + void output_c_string_with_newline(const char* str) { s << Color::None << str << "\n"; } + }; + +#ifdef DOCTEST_PLATFORM_WINDOWS + struct DebugOutputWindowReporter : public ConsoleReporter + { + DOCTEST_THREAD_LOCAL static std::ostringstream oss; + + DebugOutputWindowReporter() + : ConsoleReporter(oss) {} + +#define DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(func, type) \ + void func(type in) override { \ + if(isDebuggerActive()) { \ + bool with_col = g_no_colors; \ + g_no_colors = false; \ + ConsoleReporter::func(in); \ + DOCTEST_OUTPUT_DEBUG_STRING(oss.str().c_str()); \ + oss.str(""); \ + g_no_colors = with_col; \ + } \ + } + + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(test_run_start, const ContextOptions&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(test_run_end, const TestRunStats&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(test_case_start, const TestCaseData&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(test_case_end, const CurrentTestCaseStats&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(subcase_start, const SubcaseSignature&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(subcase_end, const SubcaseSignature&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(log_assert, const AssertData&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(log_message, const MessageData&) + DOCTEST_DEBUG_OUTPUT_WINDOW_REPORTER_OVERRIDE(test_case_skipped, const TestCaseData&) + }; + + DOCTEST_THREAD_LOCAL std::ostringstream DebugOutputWindowReporter::oss; + + DebugOutputWindowReporter g_debug_output_rep; +#endif // DOCTEST_PLATFORM_WINDOWS + + // the implementation of parseFlag() + bool parseFlagImpl(int argc, const char* const* argv, const char* pattern) { + for(int i = argc - 1; i >= 0; --i) { + auto temp = std::strstr(argv[i], pattern); + if(temp && strlen(temp) == strlen(pattern)) { + // eliminate strings in which the chars before the option are not '-' + bool noBadCharsFound = true; //!OCLINT prefer early exits and continue + while(temp != argv[i]) { + if(*--temp != '-') { + noBadCharsFound = false; + break; + } + } + if(noBadCharsFound && argv[i][0] == '-') + return true; + } + } + return false; + } + + // locates a flag on the command line + bool parseFlag(int argc, const char* const* argv, const char* pattern) { +#ifndef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS + // offset (normally 3 for "dt-") to skip prefix + if(parseFlagImpl(argc, argv, pattern + strlen(DOCTEST_CONFIG_OPTIONS_PREFIX))) + return true; +#endif // DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS + return parseFlagImpl(argc, argv, pattern); + } + + // the implementation of parseOption() + bool parseOptionImpl(int argc, const char* const* argv, const char* pattern, String& res) { + for(int i = argc - 1; i >= 0; --i) { + auto temp = std::strstr(argv[i], pattern); + if(temp) { //!OCLINT prefer early exits and continue + // eliminate matches in which the chars before the option are not '-' + bool noBadCharsFound = true; + auto curr = argv[i]; + while(curr != temp) { + if(*curr++ != '-') { + noBadCharsFound = false; + break; + } + } + if(noBadCharsFound && argv[i][0] == '-') { + temp += strlen(pattern); + const unsigned len = strlen(temp); + if(len) { + res = temp; + return true; + } + } + } + } + return false; + } + + // parses an option and returns the string after the '=' character + bool parseOption(int argc, const char* const* argv, const char* pattern, String& res, + const String& defaultVal = String()) { + res = defaultVal; +#ifndef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS + // offset (normally 3 for "dt-") to skip prefix + if(parseOptionImpl(argc, argv, pattern + strlen(DOCTEST_CONFIG_OPTIONS_PREFIX), res)) + return true; +#endif // DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS + return parseOptionImpl(argc, argv, pattern, res); + } + + // parses a comma separated list of words after a pattern in one of the arguments in argv + bool parseCommaSepArgs(int argc, const char* const* argv, const char* pattern, + std::vector& res) { + String filtersString; + if(parseOption(argc, argv, pattern, filtersString)) { + // tokenize with "," as a separator + // cppcheck-suppress strtokCalled + DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations") + auto pch = std::strtok(filtersString.c_str(), ","); // modifies the string + while(pch != nullptr) { + if(strlen(pch)) + res.push_back(pch); + // uses the strtok() internal state to go to the next token + // cppcheck-suppress strtokCalled + pch = std::strtok(nullptr, ","); + } + DOCTEST_CLANG_SUPPRESS_WARNING_POP + return true; + } + return false; + } + + enum optionType + { + option_bool, + option_int + }; + + // parses an int/bool option from the command line + bool parseIntOption(int argc, const char* const* argv, const char* pattern, optionType type, + int& res) { + String parsedValue; + if(!parseOption(argc, argv, pattern, parsedValue)) + return false; + + if(type == 0) { + // boolean + const char positive[][5] = {"1", "true", "on", "yes"}; // 5 - strlen("true") + 1 + const char negative[][6] = {"0", "false", "off", "no"}; // 6 - strlen("false") + 1 + + // if the value matches any of the positive/negative possibilities + for(unsigned i = 0; i < 4; i++) { + if(parsedValue.compare(positive[i], true) == 0) { + res = 1; //!OCLINT parameter reassignment + return true; + } + if(parsedValue.compare(negative[i], true) == 0) { + res = 0; //!OCLINT parameter reassignment + return true; + } + } + } else { + // integer + // TODO: change this to use std::stoi or something else! currently it uses undefined behavior - assumes '0' on unsuccessful parse... + int theInt = std::atoi(parsedValue.c_str()); // NOLINT + if(theInt != 0) { + res = theInt; //!OCLINT parameter reassignment + return true; + } + } + return false; + } +} // namespace + +Context::Context(int argc, const char* const* argv) + : p(new detail::ContextState) { + parseArgs(argc, argv, true); +} + +Context::~Context() { + if(g_cs == p) + g_cs = nullptr; + delete p; +} + +void Context::applyCommandLine(int argc, const char* const* argv) { parseArgs(argc, argv); } + +// parses args +void Context::parseArgs(int argc, const char* const* argv, bool withDefaults) { + using namespace detail; + + // clang-format off + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "source-file=", p->filters[0]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sf=", p->filters[0]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "source-file-exclude=",p->filters[1]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sfe=", p->filters[1]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-suite=", p->filters[2]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "ts=", p->filters[2]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-suite-exclude=", p->filters[3]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tse=", p->filters[3]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-case=", p->filters[4]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tc=", p->filters[4]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-case-exclude=", p->filters[5]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tce=", p->filters[5]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "subcase=", p->filters[6]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sc=", p->filters[6]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "subcase-exclude=", p->filters[7]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sce=", p->filters[7]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "reporters=", p->filters[8]); + parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "r=", p->filters[8]); + // clang-format on + + int intRes = 0; + String strRes; + +#define DOCTEST_PARSE_AS_BOOL_OR_FLAG(name, sname, var, default) \ + if(parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", option_bool, intRes) || \ + parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", option_bool, intRes)) \ + p->var = !!intRes; \ + else if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name) || \ + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname)) \ + p->var = true; \ + else if(withDefaults) \ + p->var = default + +#define DOCTEST_PARSE_INT_OPTION(name, sname, var, default) \ + if(parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", option_int, intRes) || \ + parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", option_int, intRes)) \ + p->var = intRes; \ + else if(withDefaults) \ + p->var = default + +#define DOCTEST_PARSE_STR_OPTION(name, sname, var, default) \ + if(parseOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", strRes, default) || \ + parseOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", strRes, default) || \ + withDefaults) \ + p->var = strRes + + // clang-format off + DOCTEST_PARSE_STR_OPTION("order-by", "ob", order_by, "file"); + DOCTEST_PARSE_INT_OPTION("rand-seed", "rs", rand_seed, 0); + + DOCTEST_PARSE_INT_OPTION("first", "f", first, 0); + DOCTEST_PARSE_INT_OPTION("last", "l", last, UINT_MAX); + + DOCTEST_PARSE_INT_OPTION("abort-after", "aa", abort_after, 0); + DOCTEST_PARSE_INT_OPTION("subcase-filter-levels", "scfl", subcase_filter_levels, 2000000000); + + DOCTEST_PARSE_AS_BOOL_OR_FLAG("success", "s", success, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("case-sensitive", "cs", case_sensitive, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("exit", "e", exit, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("duration", "d", duration, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-throw", "nt", no_throw, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-exitcode", "ne", no_exitcode, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-run", "nr", no_run, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-version", "nv", no_version, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-colors", "nc", no_colors, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("force-colors", "fc", force_colors, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-breaks", "nb", no_breaks, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-skip", "ns", no_skip, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("gnu-file-line", "gfl", gnu_file_line, !bool(DOCTEST_MSVC)); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-path-filenames", "npf", no_path_in_filenames, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-line-numbers", "nln", no_line_numbers, false); + DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-skipped-summary", "nss", no_skipped_summary, false); + // clang-format on + + if(withDefaults) { + p->help = false; + p->version = false; + p->count = false; + p->list_test_cases = false; + p->list_test_suites = false; + p->list_reporters = false; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "help") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "h") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "?")) { + p->help = true; + p->exit = true; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "version") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "v")) { + p->version = true; + p->exit = true; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "count") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "c")) { + p->count = true; + p->exit = true; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-test-cases") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "ltc")) { + p->list_test_cases = true; + p->exit = true; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-test-suites") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "lts")) { + p->list_test_suites = true; + p->exit = true; + } + if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-reporters") || + parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "lr")) { + p->list_reporters = true; + p->exit = true; + } +} + +// allows the user to add procedurally to the filters from the command line +void Context::addFilter(const char* filter, const char* value) { setOption(filter, value); } + +// allows the user to clear all filters from the command line +void Context::clearFilters() { + for(auto& curr : p->filters) + curr.clear(); +} + +// allows the user to override procedurally the int/bool options from the command line +void Context::setOption(const char* option, int value) { + setOption(option, toString(value).c_str()); +} + +// allows the user to override procedurally the string options from the command line +void Context::setOption(const char* option, const char* value) { + auto argv = String("-") + option + "=" + value; + auto lvalue = argv.c_str(); + parseArgs(1, &lvalue); +} + +// users should query this in their main() and exit the program if true +bool Context::shouldExit() { return p->exit; } + +void Context::setAsDefaultForAssertsOutOfTestCases() { g_cs = p; } + +void Context::setAssertHandler(detail::assert_handler ah) { p->ah = ah; } + +// the main function that does all the filtering and test running +int Context::run() { + using namespace detail; + + auto old_cs = g_cs; + g_cs = p; + is_running_in_test = true; + g_no_colors = p->no_colors; + p->resetRunData(); + + ConsoleReporterWithHelpers g_con_rep(std::cout); + registerReporter("console", 0, g_con_rep); + + // setup default reporter if none is given through the command line + p->reporters_currently_used.clear(); + if(p->filters[8].empty()) + p->reporters_currently_used.push_back(getReporters()[reporterMap::key_type(0, "console")]); + + // check to see if any of the registered reporters has been selected + for(auto& curr : getReporters()) { + if(matchesAny(curr.first.second.c_str(), p->filters[8], false, p->case_sensitive)) + p->reporters_currently_used.push_back(curr.second); + } + + // always use the debug output window reporter +#ifdef DOCTEST_PLATFORM_WINDOWS + if(isDebuggerActive()) + p->reporters_currently_used.push_back(&g_debug_output_rep); +#endif // DOCTEST_PLATFORM_WINDOWS + + // handle version, help and no_run + if(p->no_run || p->version || p->help || p->list_reporters) { + if(p->version) + g_con_rep.printVersion(); + if(p->help) + g_con_rep.printHelp(); + if(p->list_reporters) + g_con_rep.printRegisteredReporters(); + + g_cs = old_cs; + is_running_in_test = false; + + return EXIT_SUCCESS; + } + + g_con_rep.printIntro(); + + std::vector testArray; + for(auto& curr : getRegisteredTests()) + testArray.push_back(&curr); + p->numTestCases = testArray.size(); + + // sort the collected records + if(!testArray.empty()) { + if(p->order_by.compare("file", true) == 0) { + std::qsort(&testArray[0], testArray.size(), sizeof(TestCase*), fileOrderComparator); + } else if(p->order_by.compare("suite", true) == 0) { + std::qsort(&testArray[0], testArray.size(), sizeof(TestCase*), suiteOrderComparator); + } else if(p->order_by.compare("name", true) == 0) { + std::qsort(&testArray[0], testArray.size(), sizeof(TestCase*), nameOrderComparator); + } else if(p->order_by.compare("rand", true) == 0) { + std::srand(p->rand_seed); + + // random_shuffle implementation + const auto first = &testArray[0]; + for(size_t i = testArray.size() - 1; i > 0; --i) { + int idxToSwap = std::rand() % (i + 1); // NOLINT + + const auto temp = first[i]; + + first[i] = first[idxToSwap]; + first[idxToSwap] = temp; + } + } + } + + if(p->list_test_cases) + g_con_rep.output_query_preamble_test_cases(); + + std::set testSuitesPassingFilt; + if(p->list_test_suites) + g_con_rep.output_query_preamble_test_suites(); + + bool query_mode = p->count || p->list_test_cases || p->list_test_suites; + + if(!query_mode) + DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_start, *g_cs); + + // invoke the registered functions if they match the filter criteria (or just count them) + for(auto& curr : testArray) { + const auto tc = *curr; + + bool skip_me = false; + if(tc.m_skip && !p->no_skip) + skip_me = true; + + if(!matchesAny(tc.m_file, p->filters[0], true, p->case_sensitive)) + skip_me = true; + if(matchesAny(tc.m_file, p->filters[1], false, p->case_sensitive)) + skip_me = true; + if(!matchesAny(tc.m_test_suite, p->filters[2], true, p->case_sensitive)) + skip_me = true; + if(matchesAny(tc.m_test_suite, p->filters[3], false, p->case_sensitive)) + skip_me = true; + if(!matchesAny(tc.m_name, p->filters[4], true, p->case_sensitive)) + skip_me = true; + if(matchesAny(tc.m_name, p->filters[5], false, p->case_sensitive)) + skip_me = true; + + if(!skip_me) + p->numTestCasesPassingFilters++; + + // skip the test if it is not in the execution range + if((p->last < p->numTestCasesPassingFilters && p->first <= p->last) || + (p->first > p->numTestCasesPassingFilters)) + skip_me = true; + + if(skip_me) { + if(!query_mode) + DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_skipped, tc); + continue; + } + + // do not execute the test if we are to only count the number of filter passing tests + if(p->count) + continue; + + // print the name of the test and don't execute it + if(p->list_test_cases) { + g_con_rep.output_c_string_with_newline(tc.m_name); + continue; + } + + // print the name of the test suite if not done already and don't execute it + if(p->list_test_suites) { + if((testSuitesPassingFilt.count(tc.m_test_suite) == 0) && tc.m_test_suite[0] != '\0') { + g_con_rep.output_c_string_with_newline(tc.m_test_suite); + testSuitesPassingFilt.insert(tc.m_test_suite); + p->numTestSuitesPassingFilters++; + } + continue; + } + + // execute the test if it passes all the filtering + { + p->currentTest = &tc; + + p->failure_flags = TestCaseFailureReason::None; + p->seconds_so_far = 0; + p->error_string = ""; + + // reset non-atomic counters + p->numAssertsFailedForCurrentTestCase = 0; + p->numAssertsForCurrentTestCase = 0; + + p->subcasesPassed.clear(); + do { + // reset some of the fields for subcases (except for the set of fully passed ones) + p->should_reenter = false; + p->subcasesCurrentLevel = 0; + p->subcasesEnteredLevels.clear(); + + // reset stuff for logging with INFO() + p->stringifiedContexts.clear(); + + // reset atomic counters + p->numAssertsFailedForCurrentTestCase_atomic = 0; + p->numAssertsForCurrentTestCase_atomic = 0; + + DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_start, tc); + + g_timer.start(); + +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS + try { +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + FatalConditionHandler fatalConditionHandler; // Handle signals + // execute the test + tc.m_test(); + fatalConditionHandler.reset(); +#ifndef DOCTEST_CONFIG_NO_EXCEPTIONS + } catch(const TestFailureException&) { + p->failure_flags |= TestCaseFailureReason::AssertFailure; + } catch(...) { + p->error_string = translateActiveException(); + p->failure_flags |= TestCaseFailureReason::Exception; + } +#endif // DOCTEST_CONFIG_NO_EXCEPTIONS + + p->seconds_so_far += g_timer.getElapsedSeconds(); + + // update the non-atomic counters + p->numAsserts += p->numAssertsForCurrentTestCase_atomic; + p->numAssertsForCurrentTestCase += p->numAssertsForCurrentTestCase_atomic; + p->numAssertsFailed += p->numAssertsFailedForCurrentTestCase_atomic; + p->numAssertsFailedForCurrentTestCase += + p->numAssertsFailedForCurrentTestCase_atomic; + + // exit this loop if enough assertions have failed - even if there are more subcases + if(p->abort_after > 0 && p->numAssertsFailed >= p->abort_after) { + p->should_reenter = false; + p->failure_flags |= TestCaseFailureReason::TooManyFailedAsserts; + } + + // call it from here only if we will continue looping for other subcases and + // call it again outside of the loop for one final time - with updated flags + if(p->should_reenter == true) { + DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_end, *g_cs); + // remove these flags - it is expected that the reporters have handled these issues + p->failure_flags &= ~TestCaseFailureReason::Exception; + p->failure_flags &= ~TestCaseFailureReason::AssertFailure; + } + } while(p->should_reenter == true); + + if(p->numAssertsFailedForCurrentTestCase) + p->failure_flags |= TestCaseFailureReason::AssertFailure; + + if(Approx(p->currentTest->m_timeout).epsilon(DBL_EPSILON) != 0 && + Approx(p->seconds_so_far).epsilon(DBL_EPSILON) > p->currentTest->m_timeout) + p->failure_flags |= TestCaseFailureReason::Timeout; + + if(tc.m_should_fail) { + if(p->failure_flags) { + p->failure_flags |= TestCaseFailureReason::ShouldHaveFailedAndDid; + } else { + p->failure_flags |= TestCaseFailureReason::ShouldHaveFailedButDidnt; + } + } else if(p->failure_flags && tc.m_may_fail) { + p->failure_flags |= TestCaseFailureReason::CouldHaveFailedAndDid; + } else if(tc.m_expected_failures > 0) { + if(p->numAssertsFailedForCurrentTestCase == tc.m_expected_failures) { + p->failure_flags |= TestCaseFailureReason::FailedExactlyNumTimes; + } else { + p->failure_flags |= TestCaseFailureReason::DidntFailExactlyNumTimes; + } + } + + bool ok_to_fail = (TestCaseFailureReason::ShouldHaveFailedAndDid & p->failure_flags) || + (TestCaseFailureReason::CouldHaveFailedAndDid & p->failure_flags) || + (TestCaseFailureReason::FailedExactlyNumTimes & p->failure_flags); + + // if any subcase has failed - the whole test case has failed + if(p->failure_flags && !ok_to_fail) + p->numTestCasesFailed++; + + DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_end, *g_cs); + + p->currentTest = nullptr; + + // stop executing tests if enough assertions have failed + if(p->abort_after > 0 && p->numAssertsFailed >= p->abort_after) + break; + } + } + + if(!query_mode) + DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_end, *g_cs); + else + g_con_rep.output_query_results(); + + g_cs = old_cs; + is_running_in_test = false; + + if(p->numTestCasesFailed && !p->no_exitcode) + return EXIT_FAILURE; + return EXIT_SUCCESS; +} + +DOCTEST_DEFINE_DEFAULTS(CurrentTestCaseStats); + +DOCTEST_DEFINE_DEFAULTS(TestRunStats); + +IReporter::~IReporter() = default; + +int IReporter::get_num_active_contexts() { return detail::g_infoContexts.size(); } +const IContextScope* const* IReporter::get_active_contexts() { + return get_num_active_contexts() ? &detail::g_infoContexts[0] : nullptr; +} + +int IReporter::get_num_stringified_contexts() { return detail::g_cs->stringifiedContexts.size(); } +const String* IReporter::get_stringified_contexts() { + return get_num_stringified_contexts() ? &detail::g_cs->stringifiedContexts[0] : nullptr; +} + +int registerReporter(const char* name, int priority, IReporter& r) { + getReporters().insert(reporterMap::value_type(reporterMap::key_type(priority, name), &r)); + return 0; +} + +// see these issues on the reasoning for this: +// - https://github.com/onqtam/doctest/issues/143#issuecomment-414418903 +// - https://github.com/onqtam/doctest/issues/126 +void DOCTEST_FIX_FOR_MACOS_LIBCPP_IOSFWD_STRING_LINK_ERRORS() { std::cout << std::string(); } + +} // namespace doctest + +#endif // DOCTEST_CONFIG_DISABLE + +#ifdef DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN +int main(int argc, char** argv) { return doctest::Context(argc, argv).run(); } +#endif // DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN + +DOCTEST_CLANG_SUPPRESS_WARNING_POP +DOCTEST_MSVC_SUPPRESS_WARNING_POP +DOCTEST_GCC_SUPPRESS_WARNING_POP + +#endif // DOCTEST_LIBRARY_IMPLEMENTATION +#endif // DOCTEST_CONFIG_IMPLEMENT diff --git a/test/thirdparty/doctest/doctest_compatibility.h b/test/thirdparty/doctest/doctest_compatibility.h new file mode 100644 index 00000000..e9d035ce --- /dev/null +++ b/test/thirdparty/doctest/doctest_compatibility.h @@ -0,0 +1,35 @@ +#ifndef DOCTEST_COMPATIBILITY +#define DOCTEST_COMPATIBILITY + +#include "doctest.h" + +// Catch doesn't require a semicolon after CAPTURE but doctest does +#undef CAPTURE +#define CAPTURE(x) DOCTEST_CAPTURE(x); + +// Sections from Catch are called Subcases in doctest and don't work with std::string by default +#undef SUBCASE +#define SECTION(x) DOCTEST_SUBCASE(x) + +// convenience macro around INFO since it doesn't support temporaries (it is optimized to avoid allocations for runtime speed) +#define INFO_WITH_TEMP_IMPL(x, var_name) const auto var_name = x; INFO(var_name) // lvalue! +#define INFO_WITH_TEMP(x) INFO_WITH_TEMP_IMPL(x, DOCTEST_ANONYMOUS(DOCTEST_STD_STRING_)) + +// doctest doesn't support THROWS_WITH for std::string out of the box (has to include ...) +#define CHECK_THROWS_WITH_STD_STR_IMPL(expr, str, var_name) \ + do { \ + std::string var_name = str; \ + CHECK_THROWS_WITH(expr, var_name.c_str()); \ + } while (false) +#define CHECK_THROWS_WITH_STD_STR(expr, str) \ + CHECK_THROWS_WITH_STD_STR_IMPL(expr, str, DOCTEST_ANONYMOUS(DOCTEST_STD_STRING_)) + +// included here because for some tests in the json repository private is defined as +// public and if no STL header is included before that then in the json include when STL +// stuff is included the MSVC STL complains (errors) that C++ keywords are being redefined +#include + +// Catch does this by default +using doctest::Approx; + +#endif From 3340162efdb9c05132f8dd6bb9a9b225a892a352 Mon Sep 17 00:00:00 2001 From: onqtam Date: Tue, 15 Jan 2019 22:48:37 +0200 Subject: [PATCH 003/220] fixing osx builds - had forgotten to define this for the object file where the test runner is compiled --- test/CMakeLists.txt | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index f314ea66..63b3dd58 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -61,9 +61,13 @@ add_library(doctest_main OBJECT "src/unit.cpp" ) set_target_properties(doctest_main PROPERTIES - COMPILE_DEFINITIONS "$<$:_SCL_SECURE_NO_WARNINGS>" COMPILE_OPTIONS "$<$:/EHsc;$<$:/Od>>" ) +target_compile_definitions(doctest_main PUBLIC + "$<$:_SCL_SECURE_NO_WARNINGS>" + "DOCTEST_THREAD_LOCAL" + "DOCTEST_CONFIG_SUPER_FAST_ASSERTS" +) target_compile_features(doctest_main PUBLIC cxx_std_11) target_include_directories(doctest_main PRIVATE "thirdparty/doctest") @@ -96,10 +100,6 @@ foreach(file ${files}) string(REGEX REPLACE "unit-([^$]+)" "test-\\1" testcase ${file_basename}) add_executable(${testcase} $ ${file}) - target_compile_definitions(${testcase} PRIVATE - DOCTEST_CONFIG_SUPER_FAST_ASSERTS - DOCTEST_THREAD_LOCAL - ) target_compile_options(${testcase} PRIVATE $<$:/EHsc;$<$:/Od>> $<$>:-Wno-deprecated;-Wno-float-equal> From a2c074fd4d96b2c61d0c5d35d1e5fbee7b8b2a77 Mon Sep 17 00:00:00 2001 From: onqtam Date: Tue, 15 Jan 2019 22:58:12 +0200 Subject: [PATCH 004/220] this should really fix the XCode 6/7 builds --- test/CMakeLists.txt | 9 ++++----- test/thirdparty/doctest/doctest_compatibility.h | 1 + 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 63b3dd58..5886d7f3 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -61,13 +61,9 @@ add_library(doctest_main OBJECT "src/unit.cpp" ) set_target_properties(doctest_main PROPERTIES + COMPILE_DEFINITIONS "$<$:_SCL_SECURE_NO_WARNINGS>" COMPILE_OPTIONS "$<$:/EHsc;$<$:/Od>>" ) -target_compile_definitions(doctest_main PUBLIC - "$<$:_SCL_SECURE_NO_WARNINGS>" - "DOCTEST_THREAD_LOCAL" - "DOCTEST_CONFIG_SUPER_FAST_ASSERTS" -) target_compile_features(doctest_main PUBLIC cxx_std_11) target_include_directories(doctest_main PRIVATE "thirdparty/doctest") @@ -100,6 +96,9 @@ foreach(file ${files}) string(REGEX REPLACE "unit-([^$]+)" "test-\\1" testcase ${file_basename}) add_executable(${testcase} $ ${file}) + target_compile_definitions(${testcase} PRIVATE + DOCTEST_CONFIG_SUPER_FAST_ASSERTS + ) target_compile_options(${testcase} PRIVATE $<$:/EHsc;$<$:/Od>> $<$>:-Wno-deprecated;-Wno-float-equal> diff --git a/test/thirdparty/doctest/doctest_compatibility.h b/test/thirdparty/doctest/doctest_compatibility.h index e9d035ce..cfbe6a46 100644 --- a/test/thirdparty/doctest/doctest_compatibility.h +++ b/test/thirdparty/doctest/doctest_compatibility.h @@ -1,6 +1,7 @@ #ifndef DOCTEST_COMPATIBILITY #define DOCTEST_COMPATIBILITY +#define DOCTEST_THREAD_LOCAL // enable single-threaded builds on XCode 6/7 - https://github.com/onqtam/doctest/issues/172 #include "doctest.h" // Catch doesn't require a semicolon after CAPTURE but doctest does From 4fd9b52fc2ff796e2dcf207387f7a3b217d5e314 Mon Sep 17 00:00:00 2001 From: Ivor Wanders Date: Wed, 16 Jan 2019 08:21:49 -0500 Subject: [PATCH 005/220] Use C++11 features supported by CMake 3.1. --- CMakeLists.txt | 6 +++++- benchmarks/CMakeLists.txt | 2 +- test/CMakeLists.txt | 6 +++++- 3 files changed, 11 insertions(+), 3 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index da6f08e7..43369782 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -46,7 +46,11 @@ endif() ## add_library(${NLOHMANN_JSON_TARGET_NAME} INTERFACE) add_library(${PROJECT_NAME}::${NLOHMANN_JSON_TARGET_NAME} ALIAS ${NLOHMANN_JSON_TARGET_NAME}) -target_compile_features(${NLOHMANN_JSON_TARGET_NAME} INTERFACE cxx_std_11) +if (${CMAKE_VERSION} VERSION_LESS "3.8.0") + target_compile_features(${NLOHMANN_JSON_TARGET_NAME} INTERFACE cxx_range_for) +else() + target_compile_features(${NLOHMANN_JSON_TARGET_NAME} INTERFACE cxx_std_11) +endif() target_include_directories( ${NLOHMANN_JSON_TARGET_NAME} diff --git a/benchmarks/CMakeLists.txt b/benchmarks/CMakeLists.txt index 06da16e4..953b1cb8 100644 --- a/benchmarks/CMakeLists.txt +++ b/benchmarks/CMakeLists.txt @@ -1,4 +1,4 @@ -cmake_minimum_required(VERSION 3.1) +cmake_minimum_required(VERSION 3.8) project(JSON_Benchmarks LANGUAGES CXX) # set compiler flags diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 04acdaab..5f253bff 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -64,7 +64,11 @@ set_target_properties(catch_main PROPERTIES COMPILE_DEFINITIONS "$<$:_SCL_SECURE_NO_WARNINGS>" COMPILE_OPTIONS "$<$:/EHsc;$<$:/Od>>" ) -target_compile_features(catch_main PUBLIC cxx_std_11) +if (${CMAKE_VERSION} VERSION_LESS "3.8.0") + target_compile_features(catch_main PUBLIC cxx_range_for) +else() + target_compile_features(catch_main PUBLIC cxx_std_11) +endif() target_include_directories(catch_main PRIVATE "thirdparty/catch") # https://stackoverflow.com/questions/2368811/how-to-set-warning-level-in-cmake From a06e7f5d80af9b5a97f4a3ddcc25dd97c0b1d9dc Mon Sep 17 00:00:00 2001 From: Patrick Boettcher Date: Thu, 24 Jan 2019 16:46:51 +0100 Subject: [PATCH 006/220] JSON-pointer: add operator+() returning a new json_pointer --- include/nlohmann/detail/json_pointer.hpp | 23 +++++++++- single_include/nlohmann/json.hpp | 23 +++++++++- test/src/unit-json_pointer.cpp | 54 ++++++++++++++++++++++++ 3 files changed, 96 insertions(+), 4 deletions(-) diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 033625bc..705346c5 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -114,14 +114,33 @@ class json_pointer } /*! - @brief remove and return last reference pointer - @throw out_of_range.405 if JSON pointer has no parent + @brief append a token at the end of the reference pointer */ void push_back(const std::string& tok) { reference_tokens.push_back(tok); } + /*! + @brief append a key-token at the end of the reference pointer and return a new json-pointer. + */ + json_pointer operator+(const std::string& tok) const + { + auto ptr = *this; + ptr.push_back(tok); + return ptr; + } + + /*! + @brief append a array-index-token at the end of the reference pointer and return a new json-pointer. + */ + json_pointer operator+(const size_t& index) const + { + auto ptr = *this; + ptr.push_back(std::to_string(index)); + return ptr; + } + private: /// return whether pointer points to the root document bool is_root() const noexcept diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index ee72531b..d845f77e 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -11924,14 +11924,33 @@ class json_pointer } /*! - @brief remove and return last reference pointer - @throw out_of_range.405 if JSON pointer has no parent + @brief append a token at the end of the reference pointer */ void push_back(const std::string& tok) { reference_tokens.push_back(tok); } + /*! + @brief append a key-token at the end of the reference pointer and return a new json-pointer. + */ + json_pointer operator+(const std::string& tok) const + { + auto ptr = *this; + ptr.push_back(tok); + return ptr; + } + + /*! + @brief append a array-index-token at the end of the reference pointer and return a new json-pointer. + */ + json_pointer operator+(const size_t& index) const + { + auto ptr = *this; + ptr.push_back(std::to_string(index)); + return ptr; + } + private: /// return whether pointer points to the root document bool is_root() const noexcept diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 1bcd3938..373b2c21 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -513,4 +513,58 @@ TEST_CASE("JSON pointers") CHECK(j[ptr] == j["object"]["/"]); CHECK(ptr.to_string() == "/object/~1"); } + + SECTION("operators") + { + const json j = + { + {"", "Hello"}, + {"pi", 3.141}, + {"happy", true}, + {"name", "Niels"}, + {"nothing", nullptr}, + { + "answer", { + {"everything", 42} + } + }, + {"list", {1, 0, 2}}, + { + "object", { + {"currency", "USD"}, + {"value", 42.99}, + {"", "empty string"}, + {"/", "slash"}, + {"~", "tilde"}, + {"~1", "tilde1"} + } + } + }; + + // empty json_pointer returns the root JSON-object + auto ptr = ""_json_pointer; + CHECK(j[ptr] == j); + + // simple field access + ptr = ptr + "pi"; + CHECK(j[ptr] == j["pi"]); + + ptr.pop_back(); + CHECK(j[ptr] == j); + + // object and children access + ptr = ptr + "answer"; + ptr = ptr + "everything"; + CHECK(j[ptr] == j["answer"]["everything"]); + + ptr.pop_back(); + ptr.pop_back(); + CHECK(j[ptr] == j); + + // push key which has to be encoded + ptr = ptr + "object"; + ptr = ptr + "/"; + CHECK(j[ptr] == j["object"]["/"]); + CHECK(ptr.to_string() == "/object/~1"); + } } From 77d1d3729056c7a6d621d118ccb5ca574a7f9440 Mon Sep 17 00:00:00 2001 From: Elvis Oric Date: Wed, 30 Jan 2019 11:40:06 +0100 Subject: [PATCH 007/220] Disable installation when used as meson subproject. #1463 --- meson.build | 2 ++ 1 file changed, 2 insertions(+) diff --git a/meson.build b/meson.build index e6867e73..fb272607 100644 --- a/meson.build +++ b/meson.build @@ -12,6 +12,7 @@ nlohmann_json_multiple_headers = declare_dependency( include_directories: include_directories('include') ) +if not meson.is_subproject() install_headers('single_include/nlohmann/json.hpp', subdir: 'nlohmann') pkgc = import('pkgconfig') @@ -19,3 +20,4 @@ pkgc.generate(name: 'nlohmann_json', version: meson.project_version(), description: 'JSON for Modern C++' ) +endif From 45819dce54018985bd4e3ec4958e624df0404952 Mon Sep 17 00:00:00 2001 From: past-due <30942300+past-due@users.noreply.github.com> Date: Wed, 30 Jan 2019 12:59:50 -0500 Subject: [PATCH 008/220] Disable -Wmismatched-tags warning on tuple_size / tuple_element --- include/nlohmann/detail/iterators/iteration_proxy.hpp | 10 +++++++++- single_include/nlohmann/json.hpp | 9 +++++++++ 2 files changed, 18 insertions(+), 1 deletion(-) diff --git a/include/nlohmann/detail/iterators/iteration_proxy.hpp b/include/nlohmann/detail/iterators/iteration_proxy.hpp index a4f2a8e3..334b467b 100644 --- a/include/nlohmann/detail/iterators/iteration_proxy.hpp +++ b/include/nlohmann/detail/iterators/iteration_proxy.hpp @@ -147,6 +147,11 @@ auto get(const nlohmann::detail::iteration_proxy_value& i) -> decl // And see https://github.com/nlohmann/json/pull/1391 namespace std { +#if defined(__clang__) + // Fix: https://github.com/nlohmann/json/issues/1401 + #pragma clang diagnostic push + #pragma clang diagnostic ignored "-Wmismatched-tags" +#endif template class tuple_size<::nlohmann::detail::iteration_proxy_value> : public std::integral_constant {}; @@ -159,4 +164,7 @@ class tuple_element> get(std::declval < ::nlohmann::detail::iteration_proxy_value> ())); }; -} \ No newline at end of file +#if defined(__clang__) + #pragma clang diagnostic pop +#endif +} diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index ee72531b..b119ceac 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -1808,6 +1808,11 @@ auto get(const nlohmann::detail::iteration_proxy_value& i) -> decl // And see https://github.com/nlohmann/json/pull/1391 namespace std { +#if defined(__clang__) + // Fix: https://github.com/nlohmann/json/issues/1401 + #pragma clang diagnostic push + #pragma clang diagnostic ignored "-Wmismatched-tags" +#endif template class tuple_size<::nlohmann::detail::iteration_proxy_value> : public std::integral_constant {}; @@ -1820,8 +1825,12 @@ class tuple_element> get(std::declval < ::nlohmann::detail::iteration_proxy_value> ())); }; +#if defined(__clang__) + #pragma clang diagnostic pop +#endif } + namespace nlohmann { namespace detail From c850e9d82db4af16d995a455d8c0d20fb4c8448d Mon Sep 17 00:00:00 2001 From: garethsb-sony Date: Thu, 31 Jan 2019 19:15:36 +0000 Subject: [PATCH 009/220] Add operator/= and operator/ to construct a JSON pointer by appending two JSON pointers, as well as convenience op/= and op= to append a single unescaped token or array index; inspired by std::filesystem::path --- include/nlohmann/detail/json_pointer.hpp | 74 +++++++++++++++++------- single_include/nlohmann/json.hpp | 74 +++++++++++++++++------- test/src/unit-json_pointer.cpp | 20 +++++-- 3 files changed, 120 insertions(+), 48 deletions(-) diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 705346c5..8ec4179e 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -76,6 +76,52 @@ class json_pointer return to_string(); } + /*! + @brief append another JSON pointer at the end of this JSON pointer + */ + json_pointer& operator/=(const json_pointer& ptr) + { + reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); + return *this; + } + + /// @copydoc push_back(std::string&&) + json_pointer& operator/=(std::string token) + { + push_back(std::move(token)); + return *this; + } + + /// @copydoc operator/=(std::string) + json_pointer& operator/=(std::size_t array_index) + { + return *this /= std::to_string(array_index); + } + + /*! + @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + */ + friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) + { + return json_pointer(left_ptr) /= right_ptr; + } + + /*! + @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& ptr, std::string token) + { + return json_pointer(ptr) /= std::move(token); + } + + /*! + @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) + { + return json_pointer(lhs) /= array_index; + } + /*! @param[in] s reference token to be converted into an array index @@ -98,7 +144,7 @@ class json_pointer } /*! - @brief remove and return last reference pointer + @brief remove and return last reference token @throw out_of_range.405 if JSON pointer has no parent */ std::string pop_back() @@ -114,31 +160,17 @@ class json_pointer } /*! - @brief append a token at the end of the reference pointer + @brief append an unescaped token at the end of the reference pointer */ - void push_back(const std::string& tok) + void push_back(const std::string& token) { - reference_tokens.push_back(tok); + reference_tokens.push_back(token); } - /*! - @brief append a key-token at the end of the reference pointer and return a new json-pointer. - */ - json_pointer operator+(const std::string& tok) const + /// @copydoc push_back(const std::string&) + void push_back(std::string&& token) { - auto ptr = *this; - ptr.push_back(tok); - return ptr; - } - - /*! - @brief append a array-index-token at the end of the reference pointer and return a new json-pointer. - */ - json_pointer operator+(const size_t& index) const - { - auto ptr = *this; - ptr.push_back(std::to_string(index)); - return ptr; + reference_tokens.push_back(std::move(token)); } private: diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index d845f77e..dd7236c8 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -11886,6 +11886,52 @@ class json_pointer return to_string(); } + /*! + @brief append another JSON pointer at the end of this JSON pointer + */ + json_pointer& operator/=(const json_pointer& ptr) + { + reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); + return *this; + } + + /// @copydoc push_back(std::string&&) + json_pointer& operator/=(std::string token) + { + push_back(std::move(token)); + return *this; + } + + /// @copydoc operator/=(std::string) + json_pointer& operator/=(std::size_t array_index) + { + return *this /= std::to_string(array_index); + } + + /*! + @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + */ + friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) + { + return json_pointer(left_ptr) /= right_ptr; + } + + /*! + @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& ptr, std::string token) + { + return json_pointer(ptr) /= std::move(token); + } + + /*! + @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) + { + return json_pointer(lhs) /= array_index; + } + /*! @param[in] s reference token to be converted into an array index @@ -11908,7 +11954,7 @@ class json_pointer } /*! - @brief remove and return last reference pointer + @brief remove and return last reference token @throw out_of_range.405 if JSON pointer has no parent */ std::string pop_back() @@ -11924,31 +11970,17 @@ class json_pointer } /*! - @brief append a token at the end of the reference pointer + @brief append an unescaped token at the end of the reference pointer */ - void push_back(const std::string& tok) + void push_back(const std::string& token) { - reference_tokens.push_back(tok); + reference_tokens.push_back(token); } - /*! - @brief append a key-token at the end of the reference pointer and return a new json-pointer. - */ - json_pointer operator+(const std::string& tok) const + /// @copydoc push_back(const std::string&) + void push_back(std::string&& token) { - auto ptr = *this; - ptr.push_back(tok); - return ptr; - } - - /*! - @brief append a array-index-token at the end of the reference pointer and return a new json-pointer. - */ - json_pointer operator+(const size_t& index) const - { - auto ptr = *this; - ptr.push_back(std::to_string(index)); - return ptr; + reference_tokens.push_back(std::move(token)); } private: diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 373b2c21..b16603a7 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -499,7 +499,8 @@ TEST_CASE("JSON pointers") CHECK(j[ptr] == j); // object and children access - ptr.push_back("answer"); + const std::string answer("answer"); + ptr.push_back(answer); ptr.push_back("everything"); CHECK(j[ptr] == j["answer"]["everything"]); @@ -546,24 +547,31 @@ TEST_CASE("JSON pointers") CHECK(j[ptr] == j); // simple field access - ptr = ptr + "pi"; + ptr = ptr / "pi"; CHECK(j[ptr] == j["pi"]); ptr.pop_back(); CHECK(j[ptr] == j); // object and children access - ptr = ptr + "answer"; - ptr = ptr + "everything"; + const std::string answer("answer"); + ptr /= answer; + ptr = ptr / "everything"; CHECK(j[ptr] == j["answer"]["everything"]); ptr.pop_back(); ptr.pop_back(); CHECK(j[ptr] == j); + CHECK(ptr / ""_json_pointer == ptr); + CHECK(j["/answer"_json_pointer / "/everything"_json_pointer] == j["answer"]["everything"]); + + // list children access + CHECK(j["/list"_json_pointer / 1] == j["list"][1]); + // push key which has to be encoded - ptr = ptr + "object"; - ptr = ptr + "/"; + ptr /= "object"; + ptr = ptr / "/"; CHECK(j[ptr] == j["object"]["/"]); CHECK(ptr.to_string() == "/object/~1"); } From 5da757bbb346749dba6bc21296f8c9e84b1adbda Mon Sep 17 00:00:00 2001 From: garethsb-sony Date: Thu, 31 Jan 2019 23:39:12 +0000 Subject: [PATCH 010/220] Attempt to satisfy Coveralls by adding a test for (unchanged) operator std::string --- test/src/unit-json_pointer.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index b16603a7..765c6b48 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -438,6 +438,7 @@ TEST_CASE("JSON pointers") }) { CHECK(json::json_pointer(ptr).to_string() == ptr); + CHECK(std::string(json::json_pointer(ptr)) == ptr); } } From fb5ceb26ac1d42f88e8ad3d100c854800bd7beaf Mon Sep 17 00:00:00 2001 From: Isaac Nickaein Date: Mon, 11 Feb 2019 17:00:48 +0330 Subject: [PATCH 011/220] Fix documentation Remove two instance of duplicate paragraphs. --- include/nlohmann/json.hpp | 6 ------ single_include/nlohmann/json.hpp | 6 ------ 2 files changed, 12 deletions(-) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index d13f8a9e..9208f63c 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -6009,9 +6009,6 @@ class basic_json @pre The container storage is contiguous. Violating this precondition yields undefined behavior. **This precondition is enforced with an assertion.** - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** @warning There is no way to enforce all preconditions at compile-time. If the function is called with a noncompliant container and with @@ -6092,9 +6089,6 @@ class basic_json @pre The container storage is contiguous. Violating this precondition yields undefined behavior. **This precondition is enforced with an assertion.** - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** @warning There is no way to enforce all preconditions at compile-time. If the function is called with a noncompliant container and with diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index ee72531b..48cac19a 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -18508,9 +18508,6 @@ class basic_json @pre The container storage is contiguous. Violating this precondition yields undefined behavior. **This precondition is enforced with an assertion.** - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** @warning There is no way to enforce all preconditions at compile-time. If the function is called with a noncompliant container and with @@ -18591,9 +18588,6 @@ class basic_json @pre The container storage is contiguous. Violating this precondition yields undefined behavior. **This precondition is enforced with an assertion.** - @pre Each element of the container has a size of 1 byte. Violating this - precondition yields undefined behavior. **This precondition is enforced - with a static assertion.** @warning There is no way to enforce all preconditions at compile-time. If the function is called with a noncompliant container and with From 6a5db0095199a5b1173061e0c9477bddb1a8d788 Mon Sep 17 00:00:00 2001 From: Isaac Nickaein Date: Sun, 10 Feb 2019 16:40:30 +0330 Subject: [PATCH 012/220] Implement contains() to check existence of a key --- include/nlohmann/json.hpp | 33 ++++++++++++++++++++++++++++++++ single_include/nlohmann/json.hpp | 33 ++++++++++++++++++++++++++++++++ 2 files changed, 66 insertions(+) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index d13f8a9e..834210d8 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -3957,6 +3957,39 @@ class basic_json return is_object() ? m_value.object->count(std::forward(key)) : 0; } + /*! + @brief check the existence of an element in a JSON object + + Check whether an element exists in a JSON object with key equivalent to + @a key. If the element is not found or the JSON value is not an object, + false is returned. + + @note This method always returns false when executed on a JSON type + that is not an object. + + @param[in] key key value to check its existence. + + @return true if an element with specified @a key exists. If no such + element with such key is found or the JSON value is not an object, + false is returned. + + @complexity Logarithmic in the size of the JSON object. + + @since version 3.6.0 + */ + template + bool contains(KeyT&& key) const + { + if (is_object()) + { + return (m_value.object->find(std::forward(key)) != m_value.object->end()); + } + else + { + return false; + } + } + /// @} diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index ee72531b..aac3b16c 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -16456,6 +16456,39 @@ class basic_json return is_object() ? m_value.object->count(std::forward(key)) : 0; } + /*! + @brief check the existence of an element in a JSON object + + Check whether an element exists in a JSON object with key equivalent to + @a key. If the element is not found or the JSON value is not an object, + false is returned. + + @note This method always returns false when executed on a JSON type + that is not an object. + + @param[in] key key value to check its existence. + + @return true if an element with specified @a key exists. If no such + element with such key is found or the JSON value is not an object, + false is returned. + + @complexity Logarithmic in the size of the JSON object. + + @since version 3.6.0 + */ + template + bool contains(KeyT&& key) const + { + if (is_object()) + { + return (m_value.object->find(std::forward(key)) != m_value.object->end()); + } + else + { + return false; + } + } + /// @} From e93f3054940348e2aca777629cd3cf3f0cf3cfdf Mon Sep 17 00:00:00 2001 From: Isaac Nickaein Date: Sun, 10 Feb 2019 16:42:01 +0330 Subject: [PATCH 013/220] Add unit-test for contains() member function --- test/src/unit-element_access2.cpp | 87 +++++++++++++++++++++++++++++++ 1 file changed, 87 insertions(+) diff --git a/test/src/unit-element_access2.cpp b/test/src/unit-element_access2.cpp index 52a30073..2dc8ef94 100644 --- a/test/src/unit-element_access2.cpp +++ b/test/src/unit-element_access2.cpp @@ -982,6 +982,93 @@ TEST_CASE("element access 2") } } } + + SECTION("check existence of key in an object") + { + SECTION("existing element") + { + for (auto key : + {"integer", "unsigned", "floating", "null", "string", "boolean", "object", "array" + }) + { + CHECK(j.contains(key) == true); + CHECK(j_const.contains(key) == true); + } + } + + SECTION("nonexisting element") + { + CHECK(j.contains("foo") == false); + CHECK(j_const.contains("foo") == false); + } + + SECTION("all types") + { + SECTION("null") + { + json j_nonobject(json::value_t::null); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("string") + { + json j_nonobject(json::value_t::string); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("object") + { + json j_nonobject(json::value_t::object); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("array") + { + json j_nonobject(json::value_t::array); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("boolean") + { + json j_nonobject(json::value_t::boolean); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("number (integer)") + { + json j_nonobject(json::value_t::number_integer); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("number (unsigned)") + { + json j_nonobject(json::value_t::number_unsigned); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + + SECTION("number (floating-point)") + { + json j_nonobject(json::value_t::number_float); + const json j_nonobject_const(j_nonobject); + CHECK(j_nonobject.contains("foo") == false); + CHECK(j_nonobject_const.contains("foo") == false); + } + } + } } } From 088a2452186eb8a6277615d140e9f31586bd1b68 Mon Sep 17 00:00:00 2001 From: Isaac Nickaein Date: Fri, 22 Feb 2019 01:05:00 +0330 Subject: [PATCH 014/220] Fix typo in README.ME Fixes https://github.com/nlohmann/json/issues/1486. --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 72171069..181b6376 100644 --- a/README.md +++ b/README.md @@ -617,7 +617,7 @@ json j_patch = R"({ })"_json; // apply the patch -j_original.merge_patch(j_patch); +j_document.merge_patch(j_patch); // { // "a": "z", // "c": { From ddc9f201f45c58ff8d94e9fec5ad22873b965adc Mon Sep 17 00:00:00 2001 From: Laurent Stacul Date: Fri, 22 Feb 2019 16:05:41 +0000 Subject: [PATCH 015/220] Fix gcc9 build error test/src/unit-allocator.cpp (Issue #1472) --- test/src/unit-allocator.cpp | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/test/src/unit-allocator.cpp b/test/src/unit-allocator.cpp index 06d56d5e..29ad93f5 100644 --- a/test/src/unit-allocator.cpp +++ b/test/src/unit-allocator.cpp @@ -110,6 +110,11 @@ struct my_allocator : std::allocator p->~T(); } } + + template + struct rebind { + using other = my_allocator; + }; }; // allows deletion of raw pointer, usually hold by json_value From 164e0e54d97b445b39436ee337df64b75bbd0f60 Mon Sep 17 00:00:00 2001 From: garethsb-sony Date: Mon, 25 Feb 2019 08:57:12 +0000 Subject: [PATCH 016/220] Rename private json_pointer::is_root as public json_pointer::empty for consistency with std::filesystem::path --- include/nlohmann/detail/json_pointer.hpp | 8 ++++---- include/nlohmann/json.hpp | 2 +- single_include/nlohmann/json.hpp | 10 +++++----- 3 files changed, 10 insertions(+), 10 deletions(-) diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 8ec4179e..1efb5049 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -149,7 +149,7 @@ class json_pointer */ std::string pop_back() { - if (JSON_UNLIKELY(is_root())) + if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } @@ -173,16 +173,16 @@ class json_pointer reference_tokens.push_back(std::move(token)); } - private: /// return whether pointer points to the root document - bool is_root() const noexcept + bool empty() const noexcept { return reference_tokens.empty(); } + private: json_pointer top() const { - if (JSON_UNLIKELY(is_root())) + if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index d13f8a9e..b10c9f5b 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -7466,7 +7466,7 @@ class basic_json const auto operation_add = [&result](json_pointer & ptr, basic_json val) { // adding to the root of the target document means replacing it - if (ptr.is_root()) + if (ptr.empty()) { result = val; } diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index dd7236c8..f3688a1c 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -11959,7 +11959,7 @@ class json_pointer */ std::string pop_back() { - if (JSON_UNLIKELY(is_root())) + if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } @@ -11983,16 +11983,16 @@ class json_pointer reference_tokens.push_back(std::move(token)); } - private: /// return whether pointer points to the root document - bool is_root() const noexcept + bool empty() const noexcept { return reference_tokens.empty(); } + private: json_pointer top() const { - if (JSON_UNLIKELY(is_root())) + if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } @@ -20016,7 +20016,7 @@ class basic_json const auto operation_add = [&result](json_pointer & ptr, basic_json val) { // adding to the root of the target document means replacing it - if (ptr.is_root()) + if (ptr.empty()) { result = val; } From 08de9eeaca3b1b757586f32ff1932a62322f17e5 Mon Sep 17 00:00:00 2001 From: garethsb-sony Date: Mon, 25 Feb 2019 09:10:45 +0000 Subject: [PATCH 017/220] Add json_pointer::parent_pointer (cf. std::filesystem::path::parent_path) --- include/nlohmann/detail/json_pointer.hpp | 15 +++++++++++++++ single_include/nlohmann/json.hpp | 15 +++++++++++++++ 2 files changed, 30 insertions(+) diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 1efb5049..28a7e8f3 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -122,6 +122,21 @@ class json_pointer return json_pointer(lhs) /= array_index; } + /*! + @brief create a new JSON pointer that is the parent of this JSON pointer + */ + json_pointer parent_pointer() const + { + if (empty()) + { + return *this; + } + + json_pointer res = *this; + res.pop_back(); + return res; + } + /*! @param[in] s reference token to be converted into an array index diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index f3688a1c..15ec3b86 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -11932,6 +11932,21 @@ class json_pointer return json_pointer(lhs) /= array_index; } + /*! + @brief create a new JSON pointer that is the parent of this JSON pointer + */ + json_pointer parent_pointer() const + { + if (empty()) + { + return *this; + } + + json_pointer res = *this; + res.pop_back(); + return res; + } + /*! @param[in] s reference token to be converted into an array index From d183bd0456e9ef459b1a26179a0215f4817179e6 Mon Sep 17 00:00:00 2001 From: garethsb-sony Date: Mon, 25 Feb 2019 09:25:02 +0000 Subject: [PATCH 018/220] Tests for json_pointer::empty and json_pointer::parent_pointer --- test/src/unit-json_pointer.cpp | 19 ++++++++++++++++++- 1 file changed, 18 insertions(+), 1 deletion(-) diff --git a/test/src/unit-json_pointer.cpp b/test/src/unit-json_pointer.cpp index 765c6b48..349bdf6c 100644 --- a/test/src/unit-json_pointer.cpp +++ b/test/src/unit-json_pointer.cpp @@ -461,7 +461,7 @@ TEST_CASE("JSON pointers") } } - SECTION("push and pop") + SECTION("empty, push, pop and parent") { const json j = { @@ -490,23 +490,28 @@ TEST_CASE("JSON pointers") // empty json_pointer returns the root JSON-object auto ptr = ""_json_pointer; + CHECK(ptr.empty()); CHECK(j[ptr] == j); // simple field access ptr.push_back("pi"); + CHECK(!ptr.empty()); CHECK(j[ptr] == j["pi"]); ptr.pop_back(); + CHECK(ptr.empty()); CHECK(j[ptr] == j); // object and children access const std::string answer("answer"); ptr.push_back(answer); ptr.push_back("everything"); + CHECK(!ptr.empty()); CHECK(j[ptr] == j["answer"]["everything"]); ptr.pop_back(); ptr.pop_back(); + CHECK(ptr.empty()); CHECK(j[ptr] == j); // push key which has to be encoded @@ -514,6 +519,18 @@ TEST_CASE("JSON pointers") ptr.push_back("/"); CHECK(j[ptr] == j["object"]["/"]); CHECK(ptr.to_string() == "/object/~1"); + + CHECK(j[ptr.parent_pointer()] == j["object"]); + ptr = ptr.parent_pointer().parent_pointer(); + CHECK(ptr.empty()); + CHECK(j[ptr] == j); + // parent-pointer of the empty json_pointer is empty + ptr = ptr.parent_pointer(); + CHECK(ptr.empty()); + CHECK(j[ptr] == j); + + CHECK_THROWS_WITH(ptr.pop_back(), + "[json.exception.out_of_range.405] JSON pointer has no parent"); } SECTION("operators") From 8aeee4f7e3087ee6dd4518d1ba7cb8238fe718b7 Mon Sep 17 00:00:00 2001 From: njlr Date: Mon, 25 Feb 2019 09:44:38 +0000 Subject: [PATCH 019/220] Update README.md * Updates Buckaroo instructions to v2 --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 181b6376..a4e4ca72 100644 --- a/README.md +++ b/README.md @@ -155,7 +155,7 @@ If you are using [Spack](https://www.spack.io/) to manage your dependencies, you If you are using [hunter](https://github.com/ruslo/hunter/) on your project for external dependencies, then you can use the [nlohmann_json package](https://docs.hunter.sh/en/latest/packages/pkg/nlohmann_json.html). Please see the hunter project for any issues regarding the packaging. -If you are using [Buckaroo](https://buckaroo.pm), you can install this library's module with `buckaroo install nlohmann/json`. Please file issues [here](https://github.com/LoopPerfect/buckaroo-recipes/issues/new?title=nlohmann/nlohmann/json). +If you are using [Buckaroo](https://buckaroo.pm), you can install this library's module with `buckaroo add github.com/buckaroo-pm/nlohmann-json`. Please file issues [here](https://github.com/buckaroo-pm/nlohmann-json). There is a demo repo [here](https://github.com/njlr/buckaroo-nholmann-json-example). If you are using [vcpkg](https://github.com/Microsoft/vcpkg/) on your project for external dependencies, then you can use the [nlohmann-json package](https://github.com/Microsoft/vcpkg/tree/master/ports/nlohmann-json). Please see the vcpkg project for any issues regarding the packaging. From bb22b1003ffcf5f5a34036acca90bda3a6608bde Mon Sep 17 00:00:00 2001 From: Andreas Schwab Date: Thu, 21 Feb 2019 11:55:21 +0100 Subject: [PATCH 020/220] Do proper endian conversions This fixes all testsuite failures on big endian hosts. --- include/nlohmann/detail/input/binary_reader.hpp | 2 +- include/nlohmann/detail/output/binary_writer.hpp | 2 +- single_include/nlohmann/json.hpp | 4 ++-- test/src/unit-udt.cpp | 5 ++--- 4 files changed, 6 insertions(+), 7 deletions(-) diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index d2052234..3f173a6a 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -1853,7 +1853,7 @@ class binary_reader } // reverse byte order prior to conversion if necessary - if (is_little_endian && !InputIsLittleEndian) + if (is_little_endian != InputIsLittleEndian) { vec[sizeof(NumberType) - i - 1] = static_cast(current); } diff --git a/include/nlohmann/detail/output/binary_writer.hpp b/include/nlohmann/detail/output/binary_writer.hpp index 2f1eed4b..f191ca12 100644 --- a/include/nlohmann/detail/output/binary_writer.hpp +++ b/include/nlohmann/detail/output/binary_writer.hpp @@ -1278,7 +1278,7 @@ class binary_writer std::memcpy(vec.data(), &n, sizeof(NumberType)); // step 2: write array to output (with possible reordering) - if (is_little_endian and not OutputIsLittleEndian) + if (is_little_endian != OutputIsLittleEndian) { // reverse byte order prior to conversion if necessary std::reverse(vec.begin(), vec.end()); diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 1b83be0b..a6a7752f 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -8333,7 +8333,7 @@ class binary_reader } // reverse byte order prior to conversion if necessary - if (is_little_endian && !InputIsLittleEndian) + if (is_little_endian != InputIsLittleEndian) { vec[sizeof(NumberType) - i - 1] = static_cast(current); } @@ -9746,7 +9746,7 @@ class binary_writer std::memcpy(vec.data(), &n, sizeof(NumberType)); // step 2: write array to output (with possible reordering) - if (is_little_endian and not OutputIsLittleEndian) + if (is_little_endian != OutputIsLittleEndian) { // reverse byte order prior to conversion if necessary std::reverse(vec.begin(), vec.end()); diff --git a/test/src/unit-udt.cpp b/test/src/unit-udt.cpp index 833a4c71..0659745c 100644 --- a/test/src/unit-udt.cpp +++ b/test/src/unit-udt.cpp @@ -599,9 +599,8 @@ struct pod_serializer static void to_json(BasicJsonType& j, const T& t) noexcept { auto bytes = static_cast< const unsigned char*>(static_cast(&t)); - std::uint64_t value = bytes[0]; - for (auto i = 1; i < 8; ++i) - value |= std::uint64_t{bytes[i]} << 8 * i; + std::uint64_t value; + std::memcpy(&value, bytes, sizeof(value)); nlohmann::to_json(j, value); } }; From 7b31e56fbff1014350c0b315373679f2b76afe95 Mon Sep 17 00:00:00 2001 From: lieff Date: Mon, 25 Feb 2019 16:38:22 +0300 Subject: [PATCH 021/220] fix GCC 7.1.1 - 7.2.1 on CentOS closes https://github.com/nlohmann/json/issues/670 --- include/nlohmann/json.hpp | 2 +- single_include/nlohmann/json.hpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 3554be12..f0aff55e 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -2859,7 +2859,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value -#if defined(JSON_HAS_CPP_17) && defined(_MSC_VER) and _MSC_VER <= 1914 +#if defined(JSON_HAS_CPP_17) && (defined(__GNUC__) || (defined(_MSC_VER) and _MSC_VER <= 1914)) and not std::is_same::value #endif #endif diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 1b83be0b..d66a9b36 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -15367,7 +15367,7 @@ class basic_json #ifndef _MSC_VER // fix for issue #167 operator<< ambiguity under VS2015 and not std::is_same>::value -#if defined(JSON_HAS_CPP_17) && defined(_MSC_VER) and _MSC_VER <= 1914 +#if defined(JSON_HAS_CPP_17) && (defined(__GNUC__) || (defined(_MSC_VER) and _MSC_VER <= 1914)) and not std::is_same::value #endif #endif From 5047c7a2170709ff67ba0c1cef09252c59dcdb0b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 10 Mar 2019 22:26:57 +0100 Subject: [PATCH 022/220] :bug: added missing include #1500 --- include/nlohmann/detail/macro_scope.hpp | 1 + single_include/nlohmann/json.hpp | 1 + 2 files changed, 2 insertions(+) diff --git a/include/nlohmann/detail/macro_scope.hpp b/include/nlohmann/detail/macro_scope.hpp index b1bd1484..fdaabec6 100644 --- a/include/nlohmann/detail/macro_scope.hpp +++ b/include/nlohmann/detail/macro_scope.hpp @@ -57,6 +57,7 @@ #define JSON_CATCH(exception) catch(exception) #define JSON_INTERNAL_CATCH(exception) catch(exception) #else + #include #define JSON_THROW(exception) std::abort() #define JSON_TRY if(true) #define JSON_CATCH(exception) if(false) diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index d66a9b36..42d65e22 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -172,6 +172,7 @@ using json = basic_json<>; #define JSON_CATCH(exception) catch(exception) #define JSON_INTERNAL_CATCH(exception) catch(exception) #else + #include #define JSON_THROW(exception) std::abort() #define JSON_TRY if(true) #define JSON_CATCH(exception) if(false) From e5c7fd488d7d1e3990ba096d7225bb706e1fad04 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 10 Mar 2019 22:56:14 +0100 Subject: [PATCH 023/220] :construction_worker: trying new Travis workers --- .travis.yml | 14 +++++++++++--- README.md | 2 +- 2 files changed, 12 insertions(+), 4 deletions(-) diff --git a/.travis.yml b/.travis.yml index 4c539510..57dafc82 100644 --- a/.travis.yml +++ b/.travis.yml @@ -128,9 +128,6 @@ matrix: # OSX / Clang - - os: osx - osx_image: xcode6.4 - - os: osx osx_image: xcode7.3 @@ -164,6 +161,9 @@ matrix: - os: osx osx_image: xcode10 + - os: osx + osx_image: xcode10.1 + # Linux / GCC - os: linux @@ -290,6 +290,14 @@ matrix: sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-6.0'] packages: ['g++-6', 'clang-6.0', 'ninja-build'] + - os: linux + compiler: clang + env: COMPILER=clang++-7.0 + addons: + apt: + sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-7'] + packages: ['g++-6', 'clang-7.0', 'ninja-build'] + - os: linux compiler: clang env: diff --git a/README.md b/README.md index a4e4ca72..124f3350 100644 --- a/README.md +++ b/README.md @@ -1036,7 +1036,6 @@ The following compilers are currently used in continuous integration at [Travis] | Clang 4.0.1 | Ubuntu 14.04.1 LTS | clang version 4.0.1-svn305264-1~exp1 (branches/release_40) | | Clang 5.0.2 | Ubuntu 14.04.1 LTS | clang version 5.0.2-svn328729-1~exp1~20180509123505.100 (branches/release_50) | | Clang 6.0.1 | Ubuntu 14.04.1 LTS | clang version 6.0.1-svn334776-1~exp1~20180726133705.85 (branches/release_60) | -| Clang Xcode 6.4 | OSX 10.10.5 | Apple LLVM version 6.1.0 (clang-602.0.53) (based on LLVM 3.6.0svn) | | Clang Xcode 7.3 | OSX 10.11.6 | Apple LLVM version 7.3.0 (clang-703.0.31) | | Clang Xcode 8.0 | OSX 10.11.6 | Apple LLVM version 8.0.0 (clang-800.0.38) | | Clang Xcode 8.1 | OSX 10.12.6 | Apple LLVM version 8.0.0 (clang-800.0.42.1) | @@ -1047,6 +1046,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang Xcode 9.2 | OSX 10.13.3 | Apple LLVM version 9.1.0 (clang-902.0.39.1) | | Clang Xcode 9.3 | OSX 10.13.3 | Apple LLVM version 9.1.0 (clang-902.0.39.2) | | Clang Xcode 10.0 | OSX 10.13.3 | Apple LLVM version 10.0.0 (clang-1000.11.45.2) | +| Clang Xcode 10.1 | OSX 10.13.3 | | | Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25420.1, MSVC 19.0.24215.1 | | Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.7.180.61344, MSVC 19.14.26433.0 | From e3729ba0a5057c36d53fabda89071be64860f8fb Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 11 Mar 2019 07:44:40 +0100 Subject: [PATCH 024/220] :green_heart: fix compiler selection --- .travis.yml | 4 ++-- README.md | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index 57dafc82..aea2bd51 100644 --- a/.travis.yml +++ b/.travis.yml @@ -292,11 +292,11 @@ matrix: - os: linux compiler: clang - env: COMPILER=clang++-7.0 + env: COMPILER=clang++-7 addons: apt: sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-7'] - packages: ['g++-6', 'clang-7.0', 'ninja-build'] + packages: ['g++-6', 'clang-7', 'ninja-build'] - os: linux compiler: clang diff --git a/README.md b/README.md index 124f3350..c2a9898b 100644 --- a/README.md +++ b/README.md @@ -1046,7 +1046,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang Xcode 9.2 | OSX 10.13.3 | Apple LLVM version 9.1.0 (clang-902.0.39.1) | | Clang Xcode 9.3 | OSX 10.13.3 | Apple LLVM version 9.1.0 (clang-902.0.39.2) | | Clang Xcode 10.0 | OSX 10.13.3 | Apple LLVM version 10.0.0 (clang-1000.11.45.2) | -| Clang Xcode 10.1 | OSX 10.13.3 | | +| Clang Xcode 10.1 | OSX 10.13.3 | Apple LLVM version 10.0.0 (clang-1000.11.45.5) | | Visual Studio 14 2015 | Windows Server 2012 R2 (x64) | Microsoft (R) Build Engine version 14.0.25420.1, MSVC 19.0.24215.1 | | Visual Studio 2017 | Windows Server 2016 | Microsoft (R) Build Engine version 15.7.180.61344, MSVC 19.14.26433.0 | From 16d9cdce4535c13c7e83b842a32028fbeac90fd0 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 11 Mar 2019 15:19:50 +0100 Subject: [PATCH 025/220] :memo: updated documentation of CI --- .travis.yml | 6 +++--- README.md | 1 + 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index aea2bd51..a27302c7 100644 --- a/.travis.yml +++ b/.travis.yml @@ -301,12 +301,12 @@ matrix: - os: linux compiler: clang env: - - COMPILER=clang++-6.0 + - COMPILER=clang++-7 - CXXFLAGS=-std=c++1z addons: apt: - sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-6.0'] - packages: ['g++-6', 'clang-6.0', 'ninja-build'] + sources: ['ubuntu-toolchain-r-test', 'llvm-toolchain-trusty-7'] + packages: ['g++-6', 'clang-7', 'ninja-build'] ################ # build script # diff --git a/README.md b/README.md index c2a9898b..f0fb869b 100644 --- a/README.md +++ b/README.md @@ -1036,6 +1036,7 @@ The following compilers are currently used in continuous integration at [Travis] | Clang 4.0.1 | Ubuntu 14.04.1 LTS | clang version 4.0.1-svn305264-1~exp1 (branches/release_40) | | Clang 5.0.2 | Ubuntu 14.04.1 LTS | clang version 5.0.2-svn328729-1~exp1~20180509123505.100 (branches/release_50) | | Clang 6.0.1 | Ubuntu 14.04.1 LTS | clang version 6.0.1-svn334776-1~exp1~20180726133705.85 (branches/release_60) | +| Clang 7.0.1 | Ubuntu 14.04.1 LTS | clang version 7.0.1-svn348686-1~exp1~20181213084532.54 (branches/release_70) | | Clang Xcode 7.3 | OSX 10.11.6 | Apple LLVM version 7.3.0 (clang-703.0.31) | | Clang Xcode 8.0 | OSX 10.11.6 | Apple LLVM version 8.0.0 (clang-800.0.38) | | Clang Xcode 8.1 | OSX 10.12.6 | Apple LLVM version 8.0.0 (clang-800.0.42.1) | From 3cd1dac653a2c677af06dcaf7ae78d1298103a53 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 11 Mar 2019 15:33:23 +0100 Subject: [PATCH 026/220] :rotating_light: fix MSVC warning #1502 --- include/nlohmann/detail/input/json_sax.hpp | 20 ++++++++++---------- single_include/nlohmann/json.hpp | 20 ++++++++++---------- test/src/unit-allocator.cpp | 3 ++- 3 files changed, 22 insertions(+), 21 deletions(-) diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index 6801b3ca..4ce5a30d 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -249,16 +249,16 @@ class json_sax_dom_parser switch ((ex.id / 100) % 100) { case 1: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 4: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); // LCOV_EXCL_START case 2: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 3: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 5: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); default: assert(false); // LCOV_EXCL_STOP @@ -506,16 +506,16 @@ class json_sax_dom_callback_parser switch ((ex.id / 100) % 100) { case 1: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 4: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); // LCOV_EXCL_START case 2: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 3: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 5: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); default: assert(false); // LCOV_EXCL_STOP diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 42d65e22..2e06ef33 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -4512,16 +4512,16 @@ class json_sax_dom_parser switch ((ex.id / 100) % 100) { case 1: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 4: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); // LCOV_EXCL_START case 2: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 3: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 5: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); default: assert(false); // LCOV_EXCL_STOP @@ -4769,16 +4769,16 @@ class json_sax_dom_callback_parser switch ((ex.id / 100) % 100) { case 1: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 4: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); // LCOV_EXCL_START case 2: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 3: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); case 5: - JSON_THROW(*reinterpret_cast(&ex)); + JSON_THROW(*static_cast(&ex)); default: assert(false); // LCOV_EXCL_STOP diff --git a/test/src/unit-allocator.cpp b/test/src/unit-allocator.cpp index 29ad93f5..1c10183d 100644 --- a/test/src/unit-allocator.cpp +++ b/test/src/unit-allocator.cpp @@ -112,7 +112,8 @@ struct my_allocator : std::allocator } template - struct rebind { + struct rebind + { using other = my_allocator; }; }; From c11bead2ae1d7b3375cf14309e60069f7c75f46b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 11 Mar 2019 21:22:30 +0100 Subject: [PATCH 027/220] :construction_worker: removing more retired Travis images --- .travis.yml | 9 +++------ README.md | 2 -- 2 files changed, 3 insertions(+), 8 deletions(-) diff --git a/.travis.yml b/.travis.yml index a27302c7..4fd4a61d 100644 --- a/.travis.yml +++ b/.travis.yml @@ -128,18 +128,15 @@ matrix: # OSX / Clang + - os: osx + osx_image: xcode6.4 + - os: osx osx_image: xcode7.3 - os: osx osx_image: xcode8 - - os: osx - osx_image: xcode8.1 - - - os: osx - osx_image: xcode8.2 - - os: osx osx_image: xcode8.3 diff --git a/README.md b/README.md index f0fb869b..25d079d9 100644 --- a/README.md +++ b/README.md @@ -1039,8 +1039,6 @@ The following compilers are currently used in continuous integration at [Travis] | Clang 7.0.1 | Ubuntu 14.04.1 LTS | clang version 7.0.1-svn348686-1~exp1~20181213084532.54 (branches/release_70) | | Clang Xcode 7.3 | OSX 10.11.6 | Apple LLVM version 7.3.0 (clang-703.0.31) | | Clang Xcode 8.0 | OSX 10.11.6 | Apple LLVM version 8.0.0 (clang-800.0.38) | -| Clang Xcode 8.1 | OSX 10.12.6 | Apple LLVM version 8.0.0 (clang-800.0.42.1) | -| Clang Xcode 8.2 | OSX 10.12.6 | Apple LLVM version 8.0.0 (clang-800.0.42.1) | | Clang Xcode 8.3 | OSX 10.11.6 | Apple LLVM version 8.1.0 (clang-802.0.38) | | Clang Xcode 9.0 | OSX 10.12.6 | Apple LLVM version 9.0.0 (clang-900.0.37) | | Clang Xcode 9.1 | OSX 10.12.6 | Apple LLVM version 9.0.0 (clang-900.0.38) | From 670f42b5613a3c843270f7b53f544de53ab2cbd7 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 11 Mar 2019 22:59:22 +0100 Subject: [PATCH 028/220] :fire: removing Xcode 6.4 builder --- .travis.yml | 3 --- 1 file changed, 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index 4fd4a61d..6e9a2861 100644 --- a/.travis.yml +++ b/.travis.yml @@ -128,9 +128,6 @@ matrix: # OSX / Clang - - os: osx - osx_image: xcode6.4 - - os: osx osx_image: xcode7.3 From 546e2cbf5e95eace45ffd197a1220ce5ee551475 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Wed, 13 Mar 2019 22:14:53 +0100 Subject: [PATCH 029/220] :rotating_light: fixed some warnings --- Makefile | 1 + include/nlohmann/detail/output/serializer.hpp | 2 +- single_include/nlohmann/json.hpp | 2 +- test/src/unit-regression.cpp | 2 +- 4 files changed, 4 insertions(+), 3 deletions(-) diff --git a/Makefile b/Makefile index 297443a5..f16b209a 100644 --- a/Makefile +++ b/Makefile @@ -136,6 +136,7 @@ pedantic_clang: -Wno-float-equal \ -Wno-switch-enum -Wno-covered-switch-default \ -Wno-c++2a-compat \ + -Wno-c++17-extensions \ -Wno-padded" # calling GCC with most warnings diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index 9994372d..74c2d771 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -620,7 +620,7 @@ class serializer if (is_negative) { *buffer_ptr = '-'; - abs_value = static_cast(-1 - x) + 1; + abs_value = static_cast(std::abs(static_cast(x))); // account one more byte for the minus sign n_chars = 1 + count_digits(abs_value); diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index ded6c2bc..8f61edc7 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -11533,7 +11533,7 @@ class serializer if (is_negative) { *buffer_ptr = '-'; - abs_value = static_cast(-1 - x) + 1; + abs_value = static_cast(std::abs(static_cast(x))); // account one more byte for the minus sign n_chars = 1 + count_digits(abs_value); diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 975b9239..822cac96 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -1762,7 +1762,7 @@ TEST_CASE("regression tests") -54, -28, -26 }; std::string s; - for (int i = 0; i < sizeof(data) / sizeof(int); i++) + for (unsigned i = 0; i < sizeof(data) / sizeof(int); i++) { s += static_cast(data[i]); } From 0067ea8f9eeb0f7c3987a27e739032957e196456 Mon Sep 17 00:00:00 2001 From: Ferenc Nasztanovics Date: Wed, 13 Mar 2019 23:42:46 +0100 Subject: [PATCH 030/220] Change macros to numeric_limits #1483 --- include/nlohmann/detail/conversions/to_chars.hpp | 2 +- single_include/nlohmann/json.hpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/include/nlohmann/detail/conversions/to_chars.hpp b/include/nlohmann/detail/conversions/to_chars.hpp index 7757c95e..b6cb22ae 100644 --- a/include/nlohmann/detail/conversions/to_chars.hpp +++ b/include/nlohmann/detail/conversions/to_chars.hpp @@ -751,7 +751,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // = buffer * 10^-m + 10^-m * (1/10 * (10 * p2) ) * 2^e // = buffer * 10^-m + 10^-m * (1/10 * ((10*p2 div 2^-e) * 2^-e + (10*p2 mod 2^-e)) * 2^e // - assert(p2 <= UINT64_MAX / 10); + assert(p2 <= std::numeric_limits::max() / 10); p2 *= 10; const uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e const uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index a77bf974..50ed3068 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -10579,7 +10579,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // = buffer * 10^-m + 10^-m * (1/10 * (10 * p2) ) * 2^e // = buffer * 10^-m + 10^-m * (1/10 * ((10*p2 div 2^-e) * 2^-e + (10*p2 mod 2^-e)) * 2^e // - assert(p2 <= UINT64_MAX / 10); + assert(p2 <= std::numeric_limits::max() / 10); p2 *= 10; const uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e const uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e From 27011e37185039721a79abd4ba83d1e24acc67e8 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 15 Mar 2019 14:55:13 +0100 Subject: [PATCH 031/220] :rotating_light: fixed warnings --- .../nlohmann/detail/conversions/to_json.hpp | 2 + include/nlohmann/detail/input/json_sax.hpp | 7 +-- include/nlohmann/detail/input/lexer.hpp | 1 + include/nlohmann/detail/input/parser.hpp | 1 + include/nlohmann/detail/input/position_t.hpp | 4 +- .../detail/iterators/iteration_proxy.hpp | 2 +- .../detail/iterators/iterator_traits.hpp | 4 +- include/nlohmann/detail/json_pointer.hpp | 1 + include/nlohmann/detail/macro_scope.hpp | 2 + include/nlohmann/detail/meta/is_sax.hpp | 1 + .../nlohmann/detail/output/binary_writer.hpp | 1 + include/nlohmann/detail/output/serializer.hpp | 1 + include/nlohmann/detail/value_t.hpp | 1 + include/nlohmann/json.hpp | 17 +++---- include/nlohmann/json_fwd.hpp | 6 +-- single_include/nlohmann/json.hpp | 51 +++++++++++-------- 16 files changed, 58 insertions(+), 44 deletions(-) diff --git a/include/nlohmann/detail/conversions/to_json.hpp b/include/nlohmann/detail/conversions/to_json.hpp index ee13eb55..96095e9c 100644 --- a/include/nlohmann/detail/conversions/to_json.hpp +++ b/include/nlohmann/detail/conversions/to_json.hpp @@ -1,7 +1,9 @@ #pragma once +#include // copy #include // or, and, not #include // begin, end +#include // string #include // tuple, get #include // is_same, is_constructible, is_floating_point, is_enum, underlying_type #include // move, forward, declval, pair diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index 4ce5a30d..d267bf2b 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -1,9 +1,10 @@ #pragma once -#include +#include // assert #include -#include -#include +#include // string +#include // move +#include // vector #include #include diff --git a/include/nlohmann/detail/input/lexer.hpp b/include/nlohmann/detail/input/lexer.hpp index 6f8b32c0..fd8858b3 100644 --- a/include/nlohmann/detail/input/lexer.hpp +++ b/include/nlohmann/detail/input/lexer.hpp @@ -6,6 +6,7 @@ #include // snprintf #include // initializer_list #include // char_traits, string +#include // move #include // vector #include diff --git a/include/nlohmann/detail/input/parser.hpp b/include/nlohmann/detail/input/parser.hpp index f5744196..e1999871 100644 --- a/include/nlohmann/detail/input/parser.hpp +++ b/include/nlohmann/detail/input/parser.hpp @@ -6,6 +6,7 @@ #include // function #include // string #include // move +#include // vector #include #include diff --git a/include/nlohmann/detail/input/position_t.hpp b/include/nlohmann/detail/input/position_t.hpp index 37f4ab15..14e9649f 100644 --- a/include/nlohmann/detail/input/position_t.hpp +++ b/include/nlohmann/detail/input/position_t.hpp @@ -23,5 +23,5 @@ struct position_t } }; -} -} +} // namespace detail +} // namespace nlohmann diff --git a/include/nlohmann/detail/iterators/iteration_proxy.hpp b/include/nlohmann/detail/iterators/iteration_proxy.hpp index 334b467b..48231968 100644 --- a/include/nlohmann/detail/iterators/iteration_proxy.hpp +++ b/include/nlohmann/detail/iterators/iteration_proxy.hpp @@ -167,4 +167,4 @@ class tuple_element> #if defined(__clang__) #pragma clang diagnostic pop #endif -} +} // namespace std diff --git a/include/nlohmann/detail/iterators/iterator_traits.hpp b/include/nlohmann/detail/iterators/iterator_traits.hpp index 35ea81fb..9806be8c 100644 --- a/include/nlohmann/detail/iterators/iterator_traits.hpp +++ b/include/nlohmann/detail/iterators/iterator_traits.hpp @@ -45,5 +45,5 @@ struct iterator_traits::value>> { using pointer = T*; using reference = T&; }; -} -} +} // namespace detail +} // namespace nlohmann diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 28a7e8f3..5dba7b11 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -4,6 +4,7 @@ #include // assert #include // accumulate #include // string +#include // move #include // vector #include diff --git a/include/nlohmann/detail/macro_scope.hpp b/include/nlohmann/detail/macro_scope.hpp index fdaabec6..94d081ee 100644 --- a/include/nlohmann/detail/macro_scope.hpp +++ b/include/nlohmann/detail/macro_scope.hpp @@ -1,5 +1,7 @@ #pragma once +#include // pair + // This file contains all internal macro definitions // You MUST include macro_unscope.hpp at the end of json.hpp to undef all of them diff --git a/include/nlohmann/detail/meta/is_sax.hpp b/include/nlohmann/detail/meta/is_sax.hpp index b610beab..4a4f5bbd 100644 --- a/include/nlohmann/detail/meta/is_sax.hpp +++ b/include/nlohmann/detail/meta/is_sax.hpp @@ -2,6 +2,7 @@ #include // size_t #include // declval +#include // string #include #include diff --git a/include/nlohmann/detail/output/binary_writer.hpp b/include/nlohmann/detail/output/binary_writer.hpp index f191ca12..4d6f1577 100644 --- a/include/nlohmann/detail/output/binary_writer.hpp +++ b/include/nlohmann/detail/output/binary_writer.hpp @@ -5,6 +5,7 @@ #include // uint8_t, uint16_t, uint32_t, uint64_t #include // memcpy #include // numeric_limits +#include // string #include #include diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index 74c2d771..f51afeea 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -12,6 +12,7 @@ #include // numeric_limits #include // string #include // is_same +#include // move #include #include diff --git a/include/nlohmann/detail/value_t.hpp b/include/nlohmann/detail/value_t.hpp index 10d555b9..c4552df9 100644 --- a/include/nlohmann/detail/value_t.hpp +++ b/include/nlohmann/detail/value_t.hpp @@ -4,6 +4,7 @@ #include // and #include // size_t #include // uint8_t +#include // string namespace nlohmann { diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 50f7980f..60f453d8 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -27,8 +27,8 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#ifndef NLOHMANN_JSON_HPP -#define NLOHMANN_JSON_HPP +#ifndef INCLUDE_NLOHMANN_JSON_HPP_ +#define INCLUDE_NLOHMANN_JSON_HPP_ #define NLOHMANN_JSON_VERSION_MAJOR 3 #define NLOHMANN_JSON_VERSION_MINOR 5 @@ -42,9 +42,11 @@ SOFTWARE. #include // initializer_list #include // istream, ostream #include // random_access_iterator_tag +#include // unique_ptr #include // accumulate #include // string, stoi, to_string #include // declval, forward, move, pair, swap +#include // vector #include #include @@ -3980,14 +3982,7 @@ class basic_json template bool contains(KeyT&& key) const { - if (is_object()) - { - return (m_value.object->find(std::forward(key)) != m_value.object->end()); - } - else - { - return false; - } + return (is_object() and m_value.object->find(std::forward(key)) != m_value.object->end()); } /// @} @@ -8052,4 +8047,4 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #include -#endif +#endif // INCLUDE_NLOHMANN_JSON_HPP_ diff --git a/include/nlohmann/json_fwd.hpp b/include/nlohmann/json_fwd.hpp index 32abba91..28fd10d4 100644 --- a/include/nlohmann/json_fwd.hpp +++ b/include/nlohmann/json_fwd.hpp @@ -1,5 +1,5 @@ -#ifndef NLOHMANN_JSON_FWD_HPP -#define NLOHMANN_JSON_FWD_HPP +#ifndef INCLUDE_NLOHMANN_JSON_FWD_HPP_ +#define INCLUDE_NLOHMANN_JSON_FWD_HPP_ #include // int64_t, uint64_t #include // map @@ -61,4 +61,4 @@ uses the standard template types. using json = basic_json<>; } // namespace nlohmann -#endif +#endif // INCLUDE_NLOHMANN_JSON_FWD_HPP_ diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index a77bf974..7e228def 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -27,8 +27,8 @@ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ -#ifndef NLOHMANN_JSON_HPP -#define NLOHMANN_JSON_HPP +#ifndef INCLUDE_NLOHMANN_JSON_HPP_ +#define INCLUDE_NLOHMANN_JSON_HPP_ #define NLOHMANN_JSON_VERSION_MAJOR 3 #define NLOHMANN_JSON_VERSION_MINOR 5 @@ -42,13 +42,15 @@ SOFTWARE. #include // initializer_list #include // istream, ostream #include // random_access_iterator_tag +#include // unique_ptr #include // accumulate #include // string, stoi, to_string #include // declval, forward, move, pair, swap +#include // vector // #include -#ifndef NLOHMANN_JSON_FWD_HPP -#define NLOHMANN_JSON_FWD_HPP +#ifndef INCLUDE_NLOHMANN_JSON_FWD_HPP_ +#define INCLUDE_NLOHMANN_JSON_FWD_HPP_ #include // int64_t, uint64_t #include // map @@ -110,11 +112,13 @@ uses the standard template types. using json = basic_json<>; } // namespace nlohmann -#endif +#endif // INCLUDE_NLOHMANN_JSON_FWD_HPP_ // #include +#include // pair + // This file contains all internal macro definitions // You MUST include macro_unscope.hpp at the end of json.hpp to undef all of them @@ -403,8 +407,8 @@ struct iterator_traits::value>> using pointer = T*; using reference = T&; }; -} -} +} // namespace detail +} // namespace nlohmann // #include @@ -839,8 +843,8 @@ struct position_t } }; -} -} +} // namespace detail +} // namespace nlohmann namespace nlohmann @@ -1192,6 +1196,7 @@ class other_error : public exception #include // and #include // size_t #include // uint8_t +#include // string namespace nlohmann { @@ -1643,8 +1648,10 @@ constexpr const auto& from_json = detail::static_const::va // #include +#include // copy #include // or, and, not #include // begin, end +#include // string #include // tuple, get #include // is_same, is_constructible, is_floating_point, is_enum, underlying_type #include // move, forward, declval, pair @@ -1829,7 +1836,7 @@ class tuple_element> #if defined(__clang__) #pragma clang diagnostic pop #endif -} +} // namespace std namespace nlohmann @@ -2595,6 +2602,7 @@ class input_adapter #include // snprintf #include // initializer_list #include // char_traits, string +#include // move #include // vector // #include @@ -4106,6 +4114,7 @@ scan_number_done: #include // function #include // string #include // move +#include // vector // #include @@ -4116,6 +4125,7 @@ scan_number_done: #include // size_t #include // declval +#include // string // #include @@ -4261,10 +4271,11 @@ struct is_sax_static_asserts // #include -#include +#include // assert #include -#include -#include +#include // string +#include // move +#include // vector // #include @@ -8472,6 +8483,7 @@ class binary_reader #include // uint8_t, uint16_t, uint32_t, uint64_t #include // memcpy #include // numeric_limits +#include // string // #include @@ -9822,6 +9834,7 @@ class binary_writer #include // numeric_limits #include // string #include // is_same +#include // move // #include @@ -11821,6 +11834,7 @@ class json_ref #include // assert #include // accumulate #include // string +#include // move #include // vector // #include @@ -16555,14 +16569,7 @@ class basic_json template bool contains(KeyT&& key) const { - if (is_object()) - { - return (m_value.object->find(std::forward(key)) != m_value.object->end()); - } - else - { - return false; - } + return (is_object() and m_value.object->find(std::forward(key)) != m_value.object->end()); } /// @} @@ -20651,4 +20658,4 @@ inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std #undef NLOHMANN_BASIC_JSON_TPL -#endif +#endif // INCLUDE_NLOHMANN_JSON_HPP_ From b02ee16721f95a8066b6a1b0cbd375b46d7d1d2a Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Fri, 15 Mar 2019 15:55:52 +0100 Subject: [PATCH 032/220] :rotating_light: fixed warnings --- .../nlohmann/detail/input/binary_reader.hpp | 20 ++-- .../nlohmann/detail/input/input_adapters.hpp | 44 ++++---- include/nlohmann/detail/input/lexer.hpp | 34 +++--- include/nlohmann/detail/json_ref.hpp | 2 +- single_include/nlohmann/json.hpp | 100 +++++++++--------- 5 files changed, 100 insertions(+), 100 deletions(-) diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index 3f173a6a..370a9520 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -522,7 +522,7 @@ class binary_reader case 0x95: case 0x96: case 0x97: - return get_cbor_array(static_cast(current & 0x1F)); + return get_cbor_array(static_cast(static_cast(current) & 0x1Fu)); case 0x98: // array (one-byte uint8_t for n follows) { @@ -576,7 +576,7 @@ class binary_reader case 0xB5: case 0xB6: case 0xB7: - return get_cbor_object(static_cast(current & 0x1F)); + return get_cbor_object(static_cast(static_cast(current) & 0x1Fu)); case 0xB8: // map (one-byte uint8_t for n follows) { @@ -638,11 +638,11 @@ class binary_reader // without such support. An example of a small decoder for // half-precision floating-point numbers in the C language // is shown in Fig. 3. - const int half = (byte1 << 8) + byte2; + const unsigned int half = (byte1 << 8u) + byte2; const double val = [&half] { - const int exp = (half >> 10) & 0x1F; - const int mant = half & 0x3FF; + const unsigned int exp = (half >> 10u) & 0x1Fu; + const unsigned int mant = half & 0x3FFu; assert(0 <= exp and exp <= 32); assert(0 <= mant and mant <= 1024); switch (exp) @@ -657,7 +657,7 @@ class binary_reader return std::ldexp(mant + 1024, exp - 25); } }(); - return sax->number_float((half & 0x8000) != 0 + return sax->number_float((half & 0x8000u) != 0 ? static_cast(-val) : static_cast(val), ""); } @@ -728,7 +728,7 @@ class binary_reader case 0x76: case 0x77: { - return get_string(input_format_t::cbor, current & 0x1F, result); + return get_string(input_format_t::cbor, static_cast(current) & 0x1Fu, result); } case 0x78: // UTF-8 string (one-byte uint8_t for n follows) @@ -1026,7 +1026,7 @@ class binary_reader case 0x8D: case 0x8E: case 0x8F: - return get_msgpack_object(static_cast(current & 0x0F)); + return get_msgpack_object(static_cast(static_cast(current) & 0x0Fu)); // fixarray case 0x90: @@ -1045,7 +1045,7 @@ class binary_reader case 0x9D: case 0x9E: case 0x9F: - return get_msgpack_array(static_cast(current & 0x0F)); + return get_msgpack_array(static_cast(static_cast(current) & 0x0Fu)); // fixstr case 0xA0: @@ -1282,7 +1282,7 @@ class binary_reader case 0xBE: case 0xBF: { - return get_string(input_format_t::msgpack, current & 0x1F, result); + return get_string(input_format_t::msgpack, static_cast(current) & 0x1Fu, result); } case 0xD9: // str 8 diff --git a/include/nlohmann/detail/input/input_adapters.hpp b/include/nlohmann/detail/input/input_adapters.hpp index 8ed0a396..322e72db 100644 --- a/include/nlohmann/detail/input/input_adapters.hpp +++ b/include/nlohmann/detail/input/input_adapters.hpp @@ -165,7 +165,7 @@ struct wide_string_input_helper else { // get the current character - const auto wc = static_cast(str[current_wchar++]); + const auto wc = static_cast(str[current_wchar++]); // UTF-32 to UTF-8 encoding if (wc < 0x80) @@ -175,23 +175,23 @@ struct wide_string_input_helper } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0 | ((wc >> 6) & 0x1F); - utf8_bytes[1] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xC0u | ((wc >> 6u) & 0x1Fu); + utf8_bytes[1] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 2; } else if (wc <= 0xFFFF) { - utf8_bytes[0] = 0xE0 | ((wc >> 12) & 0x0F); - utf8_bytes[1] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[2] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xE0u | ((wc >> 12u) & 0x0Fu); + utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[2] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 3; } else if (wc <= 0x10FFFF) { - utf8_bytes[0] = 0xF0 | ((wc >> 18) & 0x07); - utf8_bytes[1] = 0x80 | ((wc >> 12) & 0x3F); - utf8_bytes[2] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[3] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xF0u | ((wc >> 18u) & 0x07u); + utf8_bytes[1] = 0x80u | ((wc >> 12u) & 0x3Fu); + utf8_bytes[2] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[3] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 4; } else @@ -220,7 +220,7 @@ struct wide_string_input_helper else { // get the current character - const auto wc = static_cast(str[current_wchar++]); + const auto wc = static_cast(str[current_wchar++]); // UTF-16 to UTF-8 encoding if (wc < 0x80) @@ -230,27 +230,27 @@ struct wide_string_input_helper } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0 | ((wc >> 6)); - utf8_bytes[1] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xC0u | ((wc >> 6u)); + utf8_bytes[1] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 2; } else if (0xD800 > wc or wc >= 0xE000) { - utf8_bytes[0] = 0xE0 | ((wc >> 12)); - utf8_bytes[1] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[2] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xE0u | ((wc >> 12u)); + utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[2] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 3; } else { if (current_wchar < str.size()) { - const auto wc2 = static_cast(str[current_wchar++]); - const int charcode = 0x10000 + (((wc & 0x3FF) << 10) | (wc2 & 0x3FF)); - utf8_bytes[0] = 0xf0 | (charcode >> 18); - utf8_bytes[1] = 0x80 | ((charcode >> 12) & 0x3F); - utf8_bytes[2] = 0x80 | ((charcode >> 6) & 0x3F); - utf8_bytes[3] = 0x80 | (charcode & 0x3F); + const auto wc2 = static_cast(str[current_wchar++]); + const auto charcode = 0x10000u + (((wc & 0x3FFu) << 10u) | (wc2 & 0x3FFu)); + utf8_bytes[0] = 0xF0u | (charcode >> 18u); + utf8_bytes[1] = 0x80u | ((charcode >> 12u) & 0x3Fu); + utf8_bytes[2] = 0x80u | ((charcode >> 6u) & 0x3Fu); + utf8_bytes[3] = 0x80u | (charcode & 0x3Fu); utf8_bytes_filled = 4; } else diff --git a/include/nlohmann/detail/input/lexer.hpp b/include/nlohmann/detail/input/lexer.hpp index fd8858b3..446ad23f 100644 --- a/include/nlohmann/detail/input/lexer.hpp +++ b/include/nlohmann/detail/input/lexer.hpp @@ -149,22 +149,22 @@ class lexer assert(current == 'u'); int codepoint = 0; - const auto factors = { 12, 8, 4, 0 }; + const auto factors = { 12u, 8u, 4u, 0u }; for (const auto factor : factors) { get(); if (current >= '0' and current <= '9') { - codepoint += ((current - 0x30) << factor); + codepoint += ((static_cast(current) - 0x30u) << factor); } else if (current >= 'A' and current <= 'F') { - codepoint += ((current - 0x37) << factor); + codepoint += ((static_cast(current) - 0x37u) << factor); } else if (current >= 'a' and current <= 'f') { - codepoint += ((current - 0x57) << factor); + codepoint += ((static_cast(current) - 0x57u) << factor); } else { @@ -324,13 +324,13 @@ class lexer // overwrite codepoint codepoint = // high surrogate occupies the most significant 22 bits - (codepoint1 << 10) + (static_cast(codepoint1) << 10u) // low surrogate occupies the least significant 15 bits - + codepoint2 + + static_cast(codepoint2) // there is still the 0xD800, 0xDC00 and 0x10000 noise // in the result so we have to subtract with: // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00; + - 0x35FDC00u; } else { @@ -365,23 +365,23 @@ class lexer else if (codepoint <= 0x7FF) { // 2-byte characters: 110xxxxx 10xxxxxx - add(0xC0 | (codepoint >> 6)); - add(0x80 | (codepoint & 0x3F)); + add(0xC0u | (static_cast(codepoint) >> 6u)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } else if (codepoint <= 0xFFFF) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - add(0xE0 | (codepoint >> 12)); - add(0x80 | ((codepoint >> 6) & 0x3F)); - add(0x80 | (codepoint & 0x3F)); + add(0xE0u | (static_cast(codepoint) >> 12u)); + add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } else { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - add(0xF0 | (codepoint >> 18)); - add(0x80 | ((codepoint >> 12) & 0x3F)); - add(0x80 | ((codepoint >> 6) & 0x3F)); - add(0x80 | (codepoint & 0x3F)); + add(0xF0u | (static_cast(codepoint) >> 18u)); + add(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu)); + add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } break; @@ -1298,7 +1298,7 @@ scan_number_done: if (JSON_LIKELY(current != std::char_traits::eof())) { - assert(token_string.size() != 0); + assert(not token_string.empty()); token_string.pop_back(); } } diff --git a/include/nlohmann/detail/json_ref.hpp b/include/nlohmann/detail/json_ref.hpp index c8dec733..26da3964 100644 --- a/include/nlohmann/detail/json_ref.hpp +++ b/include/nlohmann/detail/json_ref.hpp @@ -35,7 +35,7 @@ class json_ref is_rvalue(true) {} // class should be movable only - json_ref(json_ref&&) = default; + json_ref(json_ref&&) noexcept = default; json_ref(const json_ref&) = delete; json_ref& operator=(const json_ref&) = delete; json_ref& operator=(json_ref&&) = delete; diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 6f6b66ef..301f4ed3 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -2337,7 +2337,7 @@ struct wide_string_input_helper else { // get the current character - const auto wc = static_cast(str[current_wchar++]); + const auto wc = static_cast(str[current_wchar++]); // UTF-32 to UTF-8 encoding if (wc < 0x80) @@ -2347,23 +2347,23 @@ struct wide_string_input_helper } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0 | ((wc >> 6) & 0x1F); - utf8_bytes[1] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xC0u | ((wc >> 6u) & 0x1Fu); + utf8_bytes[1] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 2; } else if (wc <= 0xFFFF) { - utf8_bytes[0] = 0xE0 | ((wc >> 12) & 0x0F); - utf8_bytes[1] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[2] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xE0u | ((wc >> 12u) & 0x0Fu); + utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[2] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 3; } else if (wc <= 0x10FFFF) { - utf8_bytes[0] = 0xF0 | ((wc >> 18) & 0x07); - utf8_bytes[1] = 0x80 | ((wc >> 12) & 0x3F); - utf8_bytes[2] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[3] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xF0u | ((wc >> 18u) & 0x07u); + utf8_bytes[1] = 0x80u | ((wc >> 12u) & 0x3Fu); + utf8_bytes[2] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[3] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 4; } else @@ -2392,7 +2392,7 @@ struct wide_string_input_helper else { // get the current character - const auto wc = static_cast(str[current_wchar++]); + const auto wc = static_cast(str[current_wchar++]); // UTF-16 to UTF-8 encoding if (wc < 0x80) @@ -2402,27 +2402,27 @@ struct wide_string_input_helper } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0 | ((wc >> 6)); - utf8_bytes[1] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xC0u | ((wc >> 6u)); + utf8_bytes[1] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 2; } else if (0xD800 > wc or wc >= 0xE000) { - utf8_bytes[0] = 0xE0 | ((wc >> 12)); - utf8_bytes[1] = 0x80 | ((wc >> 6) & 0x3F); - utf8_bytes[2] = 0x80 | (wc & 0x3F); + utf8_bytes[0] = 0xE0u | ((wc >> 12u)); + utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); + utf8_bytes[2] = 0x80u | (wc & 0x3Fu); utf8_bytes_filled = 3; } else { if (current_wchar < str.size()) { - const auto wc2 = static_cast(str[current_wchar++]); - const int charcode = 0x10000 + (((wc & 0x3FF) << 10) | (wc2 & 0x3FF)); - utf8_bytes[0] = 0xf0 | (charcode >> 18); - utf8_bytes[1] = 0x80 | ((charcode >> 12) & 0x3F); - utf8_bytes[2] = 0x80 | ((charcode >> 6) & 0x3F); - utf8_bytes[3] = 0x80 | (charcode & 0x3F); + const auto wc2 = static_cast(str[current_wchar++]); + const auto charcode = 0x10000u + (((wc & 0x3FFu) << 10u) | (wc2 & 0x3FFu)); + utf8_bytes[0] = 0xF0u | (charcode >> 18u); + utf8_bytes[1] = 0x80u | ((charcode >> 12u) & 0x3Fu); + utf8_bytes[2] = 0x80u | ((charcode >> 6u) & 0x3Fu); + utf8_bytes[3] = 0x80u | (charcode & 0x3Fu); utf8_bytes_filled = 4; } else @@ -2748,22 +2748,22 @@ class lexer assert(current == 'u'); int codepoint = 0; - const auto factors = { 12, 8, 4, 0 }; + const auto factors = { 12u, 8u, 4u, 0u }; for (const auto factor : factors) { get(); if (current >= '0' and current <= '9') { - codepoint += ((current - 0x30) << factor); + codepoint += ((static_cast(current) - 0x30u) << factor); } else if (current >= 'A' and current <= 'F') { - codepoint += ((current - 0x37) << factor); + codepoint += ((static_cast(current) - 0x37u) << factor); } else if (current >= 'a' and current <= 'f') { - codepoint += ((current - 0x57) << factor); + codepoint += ((static_cast(current) - 0x57u) << factor); } else { @@ -2923,13 +2923,13 @@ class lexer // overwrite codepoint codepoint = // high surrogate occupies the most significant 22 bits - (codepoint1 << 10) + (static_cast(codepoint1) << 10u) // low surrogate occupies the least significant 15 bits - + codepoint2 + + static_cast(codepoint2) // there is still the 0xD800, 0xDC00 and 0x10000 noise // in the result so we have to subtract with: // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00; + - 0x35FDC00u; } else { @@ -2964,23 +2964,23 @@ class lexer else if (codepoint <= 0x7FF) { // 2-byte characters: 110xxxxx 10xxxxxx - add(0xC0 | (codepoint >> 6)); - add(0x80 | (codepoint & 0x3F)); + add(0xC0u | (static_cast(codepoint) >> 6u)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } else if (codepoint <= 0xFFFF) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - add(0xE0 | (codepoint >> 12)); - add(0x80 | ((codepoint >> 6) & 0x3F)); - add(0x80 | (codepoint & 0x3F)); + add(0xE0u | (static_cast(codepoint) >> 12u)); + add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } else { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - add(0xF0 | (codepoint >> 18)); - add(0x80 | ((codepoint >> 12) & 0x3F)); - add(0x80 | ((codepoint >> 6) & 0x3F)); - add(0x80 | (codepoint & 0x3F)); + add(0xF0u | (static_cast(codepoint) >> 18u)); + add(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu)); + add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); + add(0x80u | (static_cast(codepoint) & 0x3Fu)); } break; @@ -3897,7 +3897,7 @@ scan_number_done: if (JSON_LIKELY(current != std::char_traits::eof())) { - assert(token_string.size() != 0); + assert(not token_string.empty()); token_string.pop_back(); } } @@ -7014,7 +7014,7 @@ class binary_reader case 0x95: case 0x96: case 0x97: - return get_cbor_array(static_cast(current & 0x1F)); + return get_cbor_array(static_cast(static_cast(current) & 0x1Fu)); case 0x98: // array (one-byte uint8_t for n follows) { @@ -7068,7 +7068,7 @@ class binary_reader case 0xB5: case 0xB6: case 0xB7: - return get_cbor_object(static_cast(current & 0x1F)); + return get_cbor_object(static_cast(static_cast(current) & 0x1Fu)); case 0xB8: // map (one-byte uint8_t for n follows) { @@ -7130,11 +7130,11 @@ class binary_reader // without such support. An example of a small decoder for // half-precision floating-point numbers in the C language // is shown in Fig. 3. - const int half = (byte1 << 8) + byte2; + const unsigned int half = (byte1 << 8u) + byte2; const double val = [&half] { - const int exp = (half >> 10) & 0x1F; - const int mant = half & 0x3FF; + const unsigned int exp = (half >> 10u) & 0x1Fu; + const unsigned int mant = half & 0x3FFu; assert(0 <= exp and exp <= 32); assert(0 <= mant and mant <= 1024); switch (exp) @@ -7149,7 +7149,7 @@ class binary_reader return std::ldexp(mant + 1024, exp - 25); } }(); - return sax->number_float((half & 0x8000) != 0 + return sax->number_float((half & 0x8000u) != 0 ? static_cast(-val) : static_cast(val), ""); } @@ -7220,7 +7220,7 @@ class binary_reader case 0x76: case 0x77: { - return get_string(input_format_t::cbor, current & 0x1F, result); + return get_string(input_format_t::cbor, static_cast(current) & 0x1Fu, result); } case 0x78: // UTF-8 string (one-byte uint8_t for n follows) @@ -7518,7 +7518,7 @@ class binary_reader case 0x8D: case 0x8E: case 0x8F: - return get_msgpack_object(static_cast(current & 0x0F)); + return get_msgpack_object(static_cast(static_cast(current) & 0x0Fu)); // fixarray case 0x90: @@ -7537,7 +7537,7 @@ class binary_reader case 0x9D: case 0x9E: case 0x9F: - return get_msgpack_array(static_cast(current & 0x0F)); + return get_msgpack_array(static_cast(static_cast(current) & 0x0Fu)); // fixstr case 0xA0: @@ -7774,7 +7774,7 @@ class binary_reader case 0xBE: case 0xBF: { - return get_string(input_format_t::msgpack, current & 0x1F, result); + return get_string(input_format_t::msgpack, static_cast(current) & 0x1Fu, result); } case 0xD9: // str 8 @@ -11794,7 +11794,7 @@ class json_ref is_rvalue(true) {} // class should be movable only - json_ref(json_ref&&) = default; + json_ref(json_ref&&) noexcept = default; json_ref(const json_ref&) = delete; json_ref& operator=(const json_ref&) = delete; json_ref& operator=(json_ref&&) = delete; From 02b3494711f4aa6c55e7399452de0c97b1b2ec47 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sat, 16 Mar 2019 08:28:44 +0100 Subject: [PATCH 033/220] :fire: removing unstable macOS builds on Traivs --- .travis.yml | 6 ------ README.md | 2 -- 2 files changed, 8 deletions(-) diff --git a/.travis.yml b/.travis.yml index 6e9a2861..484c9350 100644 --- a/.travis.yml +++ b/.travis.yml @@ -128,12 +128,6 @@ matrix: # OSX / Clang - - os: osx - osx_image: xcode7.3 - - - os: osx - osx_image: xcode8 - - os: osx osx_image: xcode8.3 diff --git a/README.md b/README.md index a1c63441..8acb2842 100644 --- a/README.md +++ b/README.md @@ -1041,8 +1041,6 @@ The following compilers are currently used in continuous integration at [Travis] | Clang 5.0.2 | Ubuntu 14.04.1 LTS | clang version 5.0.2-svn328729-1~exp1~20180509123505.100 (branches/release_50) | | Clang 6.0.1 | Ubuntu 14.04.1 LTS | clang version 6.0.1-svn334776-1~exp1~20180726133705.85 (branches/release_60) | | Clang 7.0.1 | Ubuntu 14.04.1 LTS | clang version 7.0.1-svn348686-1~exp1~20181213084532.54 (branches/release_70) | -| Clang Xcode 7.3 | OSX 10.11.6 | Apple LLVM version 7.3.0 (clang-703.0.31) | -| Clang Xcode 8.0 | OSX 10.11.6 | Apple LLVM version 8.0.0 (clang-800.0.38) | | Clang Xcode 8.3 | OSX 10.11.6 | Apple LLVM version 8.1.0 (clang-802.0.38) | | Clang Xcode 9.0 | OSX 10.12.6 | Apple LLVM version 9.0.0 (clang-900.0.37) | | Clang Xcode 9.1 | OSX 10.12.6 | Apple LLVM version 9.0.0 (clang-900.0.38) | From 34f8b4f7118ce49ae3f8d62bede08d9df07dd7e2 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 00:27:44 +0100 Subject: [PATCH 034/220] :rotating_light: fixed more warnings --- .../nlohmann/detail/conversions/to_chars.hpp | 255 +++--- .../nlohmann/detail/input/binary_reader.hpp | 132 +-- .../nlohmann/detail/input/input_adapters.hpp | 57 +- include/nlohmann/detail/input/lexer.hpp | 49 +- .../nlohmann/detail/output/binary_writer.hpp | 265 +++--- include/nlohmann/detail/output/serializer.hpp | 46 +- include/nlohmann/json.hpp | 10 +- single_include/nlohmann/json.hpp | 814 +++++++++--------- 8 files changed, 818 insertions(+), 810 deletions(-) diff --git a/include/nlohmann/detail/conversions/to_chars.hpp b/include/nlohmann/detail/conversions/to_chars.hpp index b6cb22ae..198e422e 100644 --- a/include/nlohmann/detail/conversions/to_chars.hpp +++ b/include/nlohmann/detail/conversions/to_chars.hpp @@ -1,5 +1,6 @@ #pragma once +#include // array #include // assert #include // or, and, not #include // signbit, isfinite @@ -49,10 +50,10 @@ struct diyfp // f * 2^e { static constexpr int kPrecision = 64; // = q - uint64_t f = 0; + std::uint64_t f = 0; int e = 0; - constexpr diyfp(uint64_t f_, int e_) noexcept : f(f_), e(e_) {} + constexpr diyfp(std::uint64_t f_, int e_) noexcept : f(f_), e(e_) {} /*! @brief returns x - y @@ -97,23 +98,23 @@ struct diyfp // f * 2^e // // = p_lo + 2^64 p_hi - const uint64_t u_lo = x.f & 0xFFFFFFFF; - const uint64_t u_hi = x.f >> 32; - const uint64_t v_lo = y.f & 0xFFFFFFFF; - const uint64_t v_hi = y.f >> 32; + const std::uint64_t u_lo = x.f & 0xFFFFFFFFu; + const std::uint64_t u_hi = x.f >> 32u; + const std::uint64_t v_lo = y.f & 0xFFFFFFFFu; + const std::uint64_t v_hi = y.f >> 32u; - const uint64_t p0 = u_lo * v_lo; - const uint64_t p1 = u_lo * v_hi; - const uint64_t p2 = u_hi * v_lo; - const uint64_t p3 = u_hi * v_hi; + const std::uint64_t p0 = u_lo * v_lo; + const std::uint64_t p1 = u_lo * v_hi; + const std::uint64_t p2 = u_hi * v_lo; + const std::uint64_t p3 = u_hi * v_hi; - const uint64_t p0_hi = p0 >> 32; - const uint64_t p1_lo = p1 & 0xFFFFFFFF; - const uint64_t p1_hi = p1 >> 32; - const uint64_t p2_lo = p2 & 0xFFFFFFFF; - const uint64_t p2_hi = p2 >> 32; + const std::uint64_t p0_hi = p0 >> 32u; + const std::uint64_t p1_lo = p1 & 0xFFFFFFFFu; + const std::uint64_t p1_hi = p1 >> 32u; + const std::uint64_t p2_lo = p2 & 0xFFFFFFFFu; + const std::uint64_t p2_hi = p2 >> 32u; - uint64_t Q = p0_hi + p1_lo + p2_lo; + std::uint64_t Q = p0_hi + p1_lo + p2_lo; // The full product might now be computed as // @@ -124,9 +125,9 @@ struct diyfp // f * 2^e // Effectively we only need to add the highest bit in p_lo to p_hi (and // Q_hi + 1 does not overflow). - Q += uint64_t{1} << (64 - 32 - 1); // round, ties up + Q += std::uint64_t{1} << (64u - 32u - 1u); // round, ties up - const uint64_t h = p3 + p2_hi + p1_hi + (Q >> 32); + const std::uint64_t h = p3 + p2_hi + p1_hi + (Q >> 32u); return {h, x.e + y.e + 64}; } @@ -139,9 +140,9 @@ struct diyfp // f * 2^e { assert(x.f != 0); - while ((x.f >> 63) == 0) + while ((x.f >> 63u) == 0) { - x.f <<= 1; + x.f <<= 1u; x.e--; } @@ -195,13 +196,13 @@ boundaries compute_boundaries(FloatType value) constexpr int kPrecision = std::numeric_limits::digits; // = p (includes the hidden bit) constexpr int kBias = std::numeric_limits::max_exponent - 1 + (kPrecision - 1); constexpr int kMinExp = 1 - kBias; - constexpr uint64_t kHiddenBit = uint64_t{1} << (kPrecision - 1); // = 2^(p-1) + constexpr std::uint64_t kHiddenBit = std::uint64_t{1} << (kPrecision - 1); // = 2^(p-1) - using bits_type = typename std::conditional< kPrecision == 24, uint32_t, uint64_t >::type; + using bits_type = typename std::conditional::type; - const uint64_t bits = reinterpret_bits(value); - const uint64_t E = bits >> (kPrecision - 1); - const uint64_t F = bits & (kHiddenBit - 1); + const std::uint64_t bits = reinterpret_bits(value); + const std::uint64_t E = bits >> (kPrecision - 1); + const std::uint64_t F = bits & (kHiddenBit - 1); const bool is_denormal = (E == 0); const diyfp v = is_denormal @@ -304,7 +305,7 @@ constexpr int kGamma = -32; struct cached_power // c = f * 2^e ~= 10^k { - uint64_t f; + std::uint64_t f; int e; int k; }; @@ -368,91 +369,92 @@ inline cached_power get_cached_power_for_binary_exponent(int e) // NB: // Actually this function returns c, such that -60 <= e_c + e + 64 <= -34. - constexpr int kCachedPowersSize = 79; constexpr int kCachedPowersMinDecExp = -300; constexpr int kCachedPowersDecStep = 8; - static constexpr cached_power kCachedPowers[] = + static constexpr std::array kCachedPowers = { - { 0xAB70FE17C79AC6CA, -1060, -300 }, - { 0xFF77B1FCBEBCDC4F, -1034, -292 }, - { 0xBE5691EF416BD60C, -1007, -284 }, - { 0x8DD01FAD907FFC3C, -980, -276 }, - { 0xD3515C2831559A83, -954, -268 }, - { 0x9D71AC8FADA6C9B5, -927, -260 }, - { 0xEA9C227723EE8BCB, -901, -252 }, - { 0xAECC49914078536D, -874, -244 }, - { 0x823C12795DB6CE57, -847, -236 }, - { 0xC21094364DFB5637, -821, -228 }, - { 0x9096EA6F3848984F, -794, -220 }, - { 0xD77485CB25823AC7, -768, -212 }, - { 0xA086CFCD97BF97F4, -741, -204 }, - { 0xEF340A98172AACE5, -715, -196 }, - { 0xB23867FB2A35B28E, -688, -188 }, - { 0x84C8D4DFD2C63F3B, -661, -180 }, - { 0xC5DD44271AD3CDBA, -635, -172 }, - { 0x936B9FCEBB25C996, -608, -164 }, - { 0xDBAC6C247D62A584, -582, -156 }, - { 0xA3AB66580D5FDAF6, -555, -148 }, - { 0xF3E2F893DEC3F126, -529, -140 }, - { 0xB5B5ADA8AAFF80B8, -502, -132 }, - { 0x87625F056C7C4A8B, -475, -124 }, - { 0xC9BCFF6034C13053, -449, -116 }, - { 0x964E858C91BA2655, -422, -108 }, - { 0xDFF9772470297EBD, -396, -100 }, - { 0xA6DFBD9FB8E5B88F, -369, -92 }, - { 0xF8A95FCF88747D94, -343, -84 }, - { 0xB94470938FA89BCF, -316, -76 }, - { 0x8A08F0F8BF0F156B, -289, -68 }, - { 0xCDB02555653131B6, -263, -60 }, - { 0x993FE2C6D07B7FAC, -236, -52 }, - { 0xE45C10C42A2B3B06, -210, -44 }, - { 0xAA242499697392D3, -183, -36 }, - { 0xFD87B5F28300CA0E, -157, -28 }, - { 0xBCE5086492111AEB, -130, -20 }, - { 0x8CBCCC096F5088CC, -103, -12 }, - { 0xD1B71758E219652C, -77, -4 }, - { 0x9C40000000000000, -50, 4 }, - { 0xE8D4A51000000000, -24, 12 }, - { 0xAD78EBC5AC620000, 3, 20 }, - { 0x813F3978F8940984, 30, 28 }, - { 0xC097CE7BC90715B3, 56, 36 }, - { 0x8F7E32CE7BEA5C70, 83, 44 }, - { 0xD5D238A4ABE98068, 109, 52 }, - { 0x9F4F2726179A2245, 136, 60 }, - { 0xED63A231D4C4FB27, 162, 68 }, - { 0xB0DE65388CC8ADA8, 189, 76 }, - { 0x83C7088E1AAB65DB, 216, 84 }, - { 0xC45D1DF942711D9A, 242, 92 }, - { 0x924D692CA61BE758, 269, 100 }, - { 0xDA01EE641A708DEA, 295, 108 }, - { 0xA26DA3999AEF774A, 322, 116 }, - { 0xF209787BB47D6B85, 348, 124 }, - { 0xB454E4A179DD1877, 375, 132 }, - { 0x865B86925B9BC5C2, 402, 140 }, - { 0xC83553C5C8965D3D, 428, 148 }, - { 0x952AB45CFA97A0B3, 455, 156 }, - { 0xDE469FBD99A05FE3, 481, 164 }, - { 0xA59BC234DB398C25, 508, 172 }, - { 0xF6C69A72A3989F5C, 534, 180 }, - { 0xB7DCBF5354E9BECE, 561, 188 }, - { 0x88FCF317F22241E2, 588, 196 }, - { 0xCC20CE9BD35C78A5, 614, 204 }, - { 0x98165AF37B2153DF, 641, 212 }, - { 0xE2A0B5DC971F303A, 667, 220 }, - { 0xA8D9D1535CE3B396, 694, 228 }, - { 0xFB9B7CD9A4A7443C, 720, 236 }, - { 0xBB764C4CA7A44410, 747, 244 }, - { 0x8BAB8EEFB6409C1A, 774, 252 }, - { 0xD01FEF10A657842C, 800, 260 }, - { 0x9B10A4E5E9913129, 827, 268 }, - { 0xE7109BFBA19C0C9D, 853, 276 }, - { 0xAC2820D9623BF429, 880, 284 }, - { 0x80444B5E7AA7CF85, 907, 292 }, - { 0xBF21E44003ACDD2D, 933, 300 }, - { 0x8E679C2F5E44FF8F, 960, 308 }, - { 0xD433179D9C8CB841, 986, 316 }, - { 0x9E19DB92B4E31BA9, 1013, 324 }, + { + { 0xAB70FE17C79AC6CA, -1060, -300 }, + { 0xFF77B1FCBEBCDC4F, -1034, -292 }, + { 0xBE5691EF416BD60C, -1007, -284 }, + { 0x8DD01FAD907FFC3C, -980, -276 }, + { 0xD3515C2831559A83, -954, -268 }, + { 0x9D71AC8FADA6C9B5, -927, -260 }, + { 0xEA9C227723EE8BCB, -901, -252 }, + { 0xAECC49914078536D, -874, -244 }, + { 0x823C12795DB6CE57, -847, -236 }, + { 0xC21094364DFB5637, -821, -228 }, + { 0x9096EA6F3848984F, -794, -220 }, + { 0xD77485CB25823AC7, -768, -212 }, + { 0xA086CFCD97BF97F4, -741, -204 }, + { 0xEF340A98172AACE5, -715, -196 }, + { 0xB23867FB2A35B28E, -688, -188 }, + { 0x84C8D4DFD2C63F3B, -661, -180 }, + { 0xC5DD44271AD3CDBA, -635, -172 }, + { 0x936B9FCEBB25C996, -608, -164 }, + { 0xDBAC6C247D62A584, -582, -156 }, + { 0xA3AB66580D5FDAF6, -555, -148 }, + { 0xF3E2F893DEC3F126, -529, -140 }, + { 0xB5B5ADA8AAFF80B8, -502, -132 }, + { 0x87625F056C7C4A8B, -475, -124 }, + { 0xC9BCFF6034C13053, -449, -116 }, + { 0x964E858C91BA2655, -422, -108 }, + { 0xDFF9772470297EBD, -396, -100 }, + { 0xA6DFBD9FB8E5B88F, -369, -92 }, + { 0xF8A95FCF88747D94, -343, -84 }, + { 0xB94470938FA89BCF, -316, -76 }, + { 0x8A08F0F8BF0F156B, -289, -68 }, + { 0xCDB02555653131B6, -263, -60 }, + { 0x993FE2C6D07B7FAC, -236, -52 }, + { 0xE45C10C42A2B3B06, -210, -44 }, + { 0xAA242499697392D3, -183, -36 }, + { 0xFD87B5F28300CA0E, -157, -28 }, + { 0xBCE5086492111AEB, -130, -20 }, + { 0x8CBCCC096F5088CC, -103, -12 }, + { 0xD1B71758E219652C, -77, -4 }, + { 0x9C40000000000000, -50, 4 }, + { 0xE8D4A51000000000, -24, 12 }, + { 0xAD78EBC5AC620000, 3, 20 }, + { 0x813F3978F8940984, 30, 28 }, + { 0xC097CE7BC90715B3, 56, 36 }, + { 0x8F7E32CE7BEA5C70, 83, 44 }, + { 0xD5D238A4ABE98068, 109, 52 }, + { 0x9F4F2726179A2245, 136, 60 }, + { 0xED63A231D4C4FB27, 162, 68 }, + { 0xB0DE65388CC8ADA8, 189, 76 }, + { 0x83C7088E1AAB65DB, 216, 84 }, + { 0xC45D1DF942711D9A, 242, 92 }, + { 0x924D692CA61BE758, 269, 100 }, + { 0xDA01EE641A708DEA, 295, 108 }, + { 0xA26DA3999AEF774A, 322, 116 }, + { 0xF209787BB47D6B85, 348, 124 }, + { 0xB454E4A179DD1877, 375, 132 }, + { 0x865B86925B9BC5C2, 402, 140 }, + { 0xC83553C5C8965D3D, 428, 148 }, + { 0x952AB45CFA97A0B3, 455, 156 }, + { 0xDE469FBD99A05FE3, 481, 164 }, + { 0xA59BC234DB398C25, 508, 172 }, + { 0xF6C69A72A3989F5C, 534, 180 }, + { 0xB7DCBF5354E9BECE, 561, 188 }, + { 0x88FCF317F22241E2, 588, 196 }, + { 0xCC20CE9BD35C78A5, 614, 204 }, + { 0x98165AF37B2153DF, 641, 212 }, + { 0xE2A0B5DC971F303A, 667, 220 }, + { 0xA8D9D1535CE3B396, 694, 228 }, + { 0xFB9B7CD9A4A7443C, 720, 236 }, + { 0xBB764C4CA7A44410, 747, 244 }, + { 0x8BAB8EEFB6409C1A, 774, 252 }, + { 0xD01FEF10A657842C, 800, 260 }, + { 0x9B10A4E5E9913129, 827, 268 }, + { 0xE7109BFBA19C0C9D, 853, 276 }, + { 0xAC2820D9623BF429, 880, 284 }, + { 0x80444B5E7AA7CF85, 907, 292 }, + { 0xBF21E44003ACDD2D, 933, 300 }, + { 0x8E679C2F5E44FF8F, 960, 308 }, + { 0xD433179D9C8CB841, 986, 316 }, + { 0x9E19DB92B4E31BA9, 1013, 324 }, + } }; // This computation gives exactly the same results for k as @@ -466,10 +468,9 @@ inline cached_power get_cached_power_for_binary_exponent(int e) const int index = (-kCachedPowersMinDecExp + k + (kCachedPowersDecStep - 1)) / kCachedPowersDecStep; assert(index >= 0); - assert(index < kCachedPowersSize); - static_cast(kCachedPowersSize); // Fix warning. + assert(static_cast(index) < kCachedPowers.size()); - const cached_power cached = kCachedPowers[index]; + const cached_power cached = kCachedPowers[static_cast(index)]; assert(kAlpha <= cached.e + e + 64); assert(kGamma >= cached.e + e + 64); @@ -480,7 +481,7 @@ inline cached_power get_cached_power_for_binary_exponent(int e) For n != 0, returns k, such that pow10 := 10^(k-1) <= n < 10^k. For n == 0, returns 1 and sets pow10 := 1. */ -inline int find_largest_pow10(const uint32_t n, uint32_t& pow10) +inline int find_largest_pow10(const std::uint32_t n, std::uint32_t& pow10) { // LCOV_EXCL_START if (n >= 1000000000) @@ -536,8 +537,8 @@ inline int find_largest_pow10(const uint32_t n, uint32_t& pow10) } } -inline void grisu2_round(char* buf, int len, uint64_t dist, uint64_t delta, - uint64_t rest, uint64_t ten_k) +inline void grisu2_round(char* buf, int len, std::uint64_t dist, std::uint64_t delta, + std::uint64_t rest, std::uint64_t ten_k) { assert(len >= 1); assert(dist <= delta); @@ -598,8 +599,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, assert(M_plus.e >= kAlpha); assert(M_plus.e <= kGamma); - uint64_t delta = diyfp::sub(M_plus, M_minus).f; // (significand of (M+ - M-), implicit exponent is e) - uint64_t dist = diyfp::sub(M_plus, w ).f; // (significand of (M+ - w ), implicit exponent is e) + std::uint64_t delta = diyfp::sub(M_plus, M_minus).f; // (significand of (M+ - M-), implicit exponent is e) + std::uint64_t dist = diyfp::sub(M_plus, w ).f; // (significand of (M+ - w ), implicit exponent is e) // Split M+ = f * 2^e into two parts p1 and p2 (note: e < 0): // @@ -608,10 +609,10 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // = ((p1 ) * 2^-e + (p2 )) * 2^e // = p1 + p2 * 2^e - const diyfp one(uint64_t{1} << -M_plus.e, M_plus.e); + const diyfp one(std::uint64_t{1} << -M_plus.e, M_plus.e); - auto p1 = static_cast(M_plus.f >> -one.e); // p1 = f div 2^-e (Since -e >= 32, p1 fits into a 32-bit int.) - uint64_t p2 = M_plus.f & (one.f - 1); // p2 = f mod 2^-e + auto p1 = static_cast(M_plus.f >> -one.e); // p1 = f div 2^-e (Since -e >= 32, p1 fits into a 32-bit int.) + std::uint64_t p2 = M_plus.f & (one.f - 1); // p2 = f mod 2^-e // 1) // @@ -619,7 +620,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, assert(p1 > 0); - uint32_t pow10; + std::uint32_t pow10; const int k = find_largest_pow10(p1, pow10); // 10^(k-1) <= p1 < 10^k, pow10 = 10^(k-1) @@ -647,8 +648,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // M+ = buffer * 10^n + (p1 + p2 * 2^e) (buffer = 0 for n = k) // pow10 = 10^(n-1) <= p1 < 10^n // - const uint32_t d = p1 / pow10; // d = p1 div 10^(n-1) - const uint32_t r = p1 % pow10; // r = p1 mod 10^(n-1) + const std::uint32_t d = p1 / pow10; // d = p1 div 10^(n-1) + const std::uint32_t r = p1 % pow10; // r = p1 mod 10^(n-1) // // M+ = buffer * 10^n + (d * 10^(n-1) + r) + p2 * 2^e // = (buffer * 10 + d) * 10^(n-1) + (r + p2 * 2^e) @@ -673,7 +674,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // Note: // Since rest and delta share the same exponent e, it suffices to // compare the significands. - const uint64_t rest = (uint64_t{p1} << -one.e) + p2; + const std::uint64_t rest = (std::uint64_t{p1} << -one.e) + p2; if (rest <= delta) { // V = buffer * 10^n, with M- <= V <= M+. @@ -689,7 +690,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // // 10^n = (10^n * 2^-e) * 2^e = ulp * 2^e // - const uint64_t ten_n = uint64_t{pow10} << -one.e; + const std::uint64_t ten_n = std::uint64_t{pow10} << -one.e; grisu2_round(buffer, length, dist, delta, rest, ten_n); return; @@ -753,8 +754,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // assert(p2 <= std::numeric_limits::max() / 10); p2 *= 10; - const uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e - const uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e + const std::uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e + const std::uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e // // M+ = buffer * 10^-m + 10^-m * (1/10 * (d * 2^-e + r) * 2^e // = buffer * 10^-m + 10^-m * (1/10 * (d + r * 2^e)) @@ -794,7 +795,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // // 10^m * 10^-m = 1 = 2^-e * 2^e = ten_m * 2^e // - const uint64_t ten_m = one.f; + const std::uint64_t ten_m = one.f; grisu2_round(buffer, length, dist, delta, p2, ten_m); // By construction this algorithm generates the shortest possible decimal @@ -929,7 +930,7 @@ inline char* append_exponent(char* buf, int e) *buf++ = '+'; } - auto k = static_cast(e); + auto k = static_cast(e); if (k < 10) { // Always print at least two digits in the exponent. diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index 370a9520..ff6b1ab6 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -263,9 +263,9 @@ class binary_reader default: // anything else not supported (yet) { - char cr[3]; - (std::snprintf)(cr, sizeof(cr), "%.2hhX", static_cast(element_type)); - return sax->parse_error(element_type_parse_position, std::string(cr), parse_error::create(114, element_type_parse_position, "Unsupported BSON record type 0x" + std::string(cr))); + std::array cr{}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(element_type)); + return sax->parse_error(element_type_parse_position, std::string(cr.data()), parse_error::create(114, element_type_parse_position, "Unsupported BSON record type 0x" + std::string(cr.data()))); } } } @@ -388,25 +388,25 @@ class binary_reader case 0x18: // Unsigned integer (one-byte uint8_t follows) { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x19: // Unsigned integer (two-byte uint16_t follows) { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x1A: // Unsigned integer (four-byte uint32_t follows) { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x1B: // Unsigned integer (eight-byte uint64_t follows) { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } @@ -435,29 +435,29 @@ class binary_reader case 0x35: case 0x36: case 0x37: - return sax->number_integer(static_cast(0x20 - 1 - current)); + return sax->number_integer(static_cast(0x20 - 1 - current)); case 0x38: // Negative integer (one-byte uint8_t follows) { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x39: // Negative integer -1-n (two-byte uint16_t follows) { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x3A: // Negative integer -1-n (four-byte uint32_t follows) { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x3B: // Negative integer -1-n (eight-byte uint64_t follows) { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - static_cast(number)); } @@ -526,25 +526,25 @@ class binary_reader case 0x98: // array (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x99: // array (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x9A: // array (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x9B: // array (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } @@ -580,25 +580,25 @@ class binary_reader case 0xB8: // map (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xB9: // map (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xBA: // map (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xBB: // map (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } @@ -638,10 +638,10 @@ class binary_reader // without such support. An example of a small decoder for // half-precision floating-point numbers in the C language // is shown in Fig. 3. - const unsigned int half = (byte1 << 8u) + byte2; + const auto half = static_cast((byte1 << 8u) + byte2); const double val = [&half] { - const unsigned int exp = (half >> 10u) & 0x1Fu; + const int exp = (half >> 10u) & 0x1Fu; const unsigned int mant = half & 0x3FFu; assert(0 <= exp and exp <= 32); assert(0 <= mant and mant <= 1024); @@ -733,25 +733,25 @@ class binary_reader case 0x78: // UTF-8 string (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x79: // UTF-8 string (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x7A: // UTF-8 string (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x7B: // UTF-8 string (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } @@ -1108,49 +1108,49 @@ class binary_reader case 0xCC: // uint 8 { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCD: // uint 16 { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCE: // uint 32 { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCF: // uint 64 { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xD0: // int 8 { - int8_t number; + std::int8_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD1: // int 16 { - int16_t number; + std::int16_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD2: // int 32 { - int32_t number; + std::int32_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD3: // int 64 { - int64_t number; + std::int64_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } @@ -1164,25 +1164,25 @@ class binary_reader case 0xDC: // array 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_array(static_cast(len)); } case 0xDD: // array 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_array(static_cast(len)); } case 0xDE: // map 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_object(static_cast(len)); } case 0xDF: // map 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_object(static_cast(len)); } @@ -1219,7 +1219,7 @@ class binary_reader case 0xFD: case 0xFE: case 0xFF: - return sax->number_integer(static_cast(current)); + return sax->number_integer(static_cast(current)); default: // anything else { @@ -1287,19 +1287,19 @@ class binary_reader case 0xD9: // str 8 { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } case 0xDA: // str 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } case 0xDB: // str 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } @@ -1397,7 +1397,7 @@ class binary_reader { if (get_char) { - get(); // TODO: may we ignore N here? + get(); // TODO(niels): may we ignore N here? } if (JSON_UNLIKELY(not unexpect_eof(input_format_t::ubjson, "value"))) @@ -1409,31 +1409,31 @@ class binary_reader { case 'U': { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'i': { - int8_t len; + std::int8_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'I': { - int16_t len; + std::int16_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'l': { - int32_t len; + std::int32_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'L': { - int64_t len; + std::int64_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } @@ -1453,7 +1453,7 @@ class binary_reader { case 'U': { - uint8_t number; + std::uint8_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -1464,7 +1464,7 @@ class binary_reader case 'i': { - int8_t number; + std::int8_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -1475,7 +1475,7 @@ class binary_reader case 'I': { - int16_t number; + std::int16_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -1486,7 +1486,7 @@ class binary_reader case 'l': { - int32_t number; + std::int32_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -1497,7 +1497,7 @@ class binary_reader case 'L': { - int64_t number; + std::int64_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -1552,10 +1552,12 @@ class binary_reader return get_ubjson_size_value(result.first); } - else if (current == '#') + + if (current == '#') { return get_ubjson_size_value(result.first); } + return true; } @@ -1580,31 +1582,31 @@ class binary_reader case 'U': { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::ubjson, number) and sax->number_unsigned(number); } case 'i': { - int8_t number; + std::int8_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'I': { - int16_t number; + std::int16_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'l': { - int32_t number; + std::int32_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'L': { - int64_t number; + std::int64_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } @@ -1843,7 +1845,7 @@ class binary_reader bool get_number(const input_format_t format, NumberType& result) { // step 1: read input into array with system's byte order - std::array vec; + std::array vec; for (std::size_t i = 0; i < sizeof(NumberType); ++i) { get(); @@ -1855,11 +1857,11 @@ class binary_reader // reverse byte order prior to conversion if necessary if (is_little_endian != InputIsLittleEndian) { - vec[sizeof(NumberType) - i - 1] = static_cast(current); + vec[sizeof(NumberType) - i - 1] = static_cast(current); } else { - vec[i] = static_cast(current); // LCOV_EXCL_LINE + vec[i] = static_cast(current); // LCOV_EXCL_LINE } } @@ -1920,9 +1922,9 @@ class binary_reader */ std::string get_token_string() const { - char cr[3]; - (std::snprintf)(cr, 3, "%.2hhX", static_cast(current)); - return std::string{cr}; + std::array cr{}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(current)); + return std::string{cr.data()}; } /*! diff --git a/include/nlohmann/detail/input/input_adapters.hpp b/include/nlohmann/detail/input/input_adapters.hpp index 322e72db..c3d33391 100644 --- a/include/nlohmann/detail/input/input_adapters.hpp +++ b/include/nlohmann/detail/input/input_adapters.hpp @@ -1,5 +1,6 @@ #pragma once +#include // array #include // assert #include // size_t #include // strlen @@ -153,7 +154,11 @@ template struct wide_string_input_helper { // UTF-32 - static void fill_buffer(const WideStringType& str, size_t& current_wchar, std::array::int_type, 4>& utf8_bytes, size_t& utf8_bytes_index, size_t& utf8_bytes_filled) + static void fill_buffer(const WideStringType& str, + size_t& current_wchar, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) { utf8_bytes_index = 0; @@ -170,34 +175,34 @@ struct wide_string_input_helper // UTF-32 to UTF-8 encoding if (wc < 0x80) { - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0u | ((wc >> 6u) & 0x1Fu); - utf8_bytes[1] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((wc >> 6u) & 0x1Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 2; } else if (wc <= 0xFFFF) { - utf8_bytes[0] = 0xE0u | ((wc >> 12u) & 0x0Fu); - utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[2] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((wc >> 12u) & 0x0Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 3; } else if (wc <= 0x10FFFF) { - utf8_bytes[0] = 0xF0u | ((wc >> 18u) & 0x07u); - utf8_bytes[1] = 0x80u | ((wc >> 12u) & 0x3Fu); - utf8_bytes[2] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[3] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xF0u | ((wc >> 18u) & 0x07u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 4; } else { // unknown character - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } } @@ -208,7 +213,11 @@ template struct wide_string_input_helper { // UTF-16 - static void fill_buffer(const WideStringType& str, size_t& current_wchar, std::array::int_type, 4>& utf8_bytes, size_t& utf8_bytes_index, size_t& utf8_bytes_filled) + static void fill_buffer(const WideStringType& str, + size_t& current_wchar, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) { utf8_bytes_index = 0; @@ -225,20 +234,20 @@ struct wide_string_input_helper // UTF-16 to UTF-8 encoding if (wc < 0x80) { - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0u | ((wc >> 6u)); - utf8_bytes[1] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((wc >> 6u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 2; } else if (0xD800 > wc or wc >= 0xE000) { - utf8_bytes[0] = 0xE0u | ((wc >> 12u)); - utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[2] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((wc >> 12u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 3; } else @@ -247,17 +256,17 @@ struct wide_string_input_helper { const auto wc2 = static_cast(str[current_wchar++]); const auto charcode = 0x10000u + (((wc & 0x3FFu) << 10u) | (wc2 & 0x3FFu)); - utf8_bytes[0] = 0xF0u | (charcode >> 18u); - utf8_bytes[1] = 0x80u | ((charcode >> 12u) & 0x3Fu); - utf8_bytes[2] = 0x80u | ((charcode >> 6u) & 0x3Fu); - utf8_bytes[3] = 0x80u | (charcode & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xF0u | (charcode >> 18u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((charcode >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((charcode >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (charcode & 0x3Fu)); utf8_bytes_filled = 4; } else { // unknown character ++current_wchar; - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } } diff --git a/include/nlohmann/detail/input/lexer.hpp b/include/nlohmann/detail/input/lexer.hpp index 446ad23f..a6bde97e 100644 --- a/include/nlohmann/detail/input/lexer.hpp +++ b/include/nlohmann/detail/input/lexer.hpp @@ -1,5 +1,6 @@ #pragma once +#include // array #include // localeconv #include // size_t #include // strtof, strtod, strtold, strtoll, strtoull @@ -156,15 +157,15 @@ class lexer if (current >= '0' and current <= '9') { - codepoint += ((static_cast(current) - 0x30u) << factor); + codepoint += static_cast((static_cast(current) - 0x30u) << factor); } else if (current >= 'A' and current <= 'F') { - codepoint += ((static_cast(current) - 0x37u) << factor); + codepoint += static_cast((static_cast(current) - 0x37u) << factor); } else if (current >= 'a' and current <= 'f') { - codepoint += ((static_cast(current) - 0x57u) << factor); + codepoint += static_cast((static_cast(current) - 0x57u) << factor); } else { @@ -322,15 +323,15 @@ class lexer if (JSON_LIKELY(0xDC00 <= codepoint2 and codepoint2 <= 0xDFFF)) { // overwrite codepoint - codepoint = - // high surrogate occupies the most significant 22 bits - (static_cast(codepoint1) << 10u) - // low surrogate occupies the least significant 15 bits - + static_cast(codepoint2) - // there is still the 0xD800, 0xDC00 and 0x10000 noise - // in the result so we have to subtract with: - // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00u; + codepoint = static_cast( + // high surrogate occupies the most significant 22 bits + (static_cast(codepoint1) << 10u) + // low surrogate occupies the least significant 15 bits + + static_cast(codepoint2) + // there is still the 0xD800, 0xDC00 and 0x10000 noise + // in the result so we have to subtract with: + // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 + - 0x35FDC00u); } else { @@ -365,23 +366,23 @@ class lexer else if (codepoint <= 0x7FF) { // 2-byte characters: 110xxxxx 10xxxxxx - add(0xC0u | (static_cast(codepoint) >> 6u)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xC0u | (static_cast(codepoint) >> 6u))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } else if (codepoint <= 0xFFFF) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - add(0xE0u | (static_cast(codepoint) >> 12u)); - add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xE0u | (static_cast(codepoint) >> 12u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } else { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - add(0xF0u | (static_cast(codepoint) >> 18u)); - add(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu)); - add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xF0u | (static_cast(codepoint) >> 18u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } break; @@ -1360,9 +1361,9 @@ scan_number_done: if ('\x00' <= c and c <= '\x1F') { // escape control characters - char cs[9]; - (std::snprintf)(cs, 9, "", static_cast(c)); - result += cs; + std::array cs{}; + (std::snprintf)(cs.data(), cs.size(), "", static_cast(c)); + result += cs.data(); } else { diff --git a/include/nlohmann/detail/output/binary_writer.hpp b/include/nlohmann/detail/output/binary_writer.hpp index 4d6f1577..a8fddfee 100644 --- a/include/nlohmann/detail/output/binary_writer.hpp +++ b/include/nlohmann/detail/output/binary_writer.hpp @@ -88,27 +88,27 @@ class binary_writer // code from the value_t::number_unsigned case here. if (j.m_value.number_integer <= 0x17) { - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x18)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x19)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x1A)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } else { oa->write_character(to_char_type(0x1B)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } else @@ -118,27 +118,27 @@ class binary_writer const auto positive_number = -1 - j.m_value.number_integer; if (j.m_value.number_integer >= -24) { - write_number(static_cast(0x20 + positive_number)); + write_number(static_cast(0x20 + positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x38)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x39)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x3A)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } else { oa->write_character(to_char_type(0x3B)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } } break; @@ -148,27 +148,27 @@ class binary_writer { if (j.m_value.number_unsigned <= 0x17) { - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x18)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x19)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x1A)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } else { oa->write_character(to_char_type(0x1B)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } break; } @@ -186,28 +186,28 @@ class binary_writer const auto N = j.m_value.string->size(); if (N <= 0x17) { - write_number(static_cast(0x60 + N)); + write_number(static_cast(0x60 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x78)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x79)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x7A)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x7B)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -224,28 +224,28 @@ class binary_writer const auto N = j.m_value.array->size(); if (N <= 0x17) { - write_number(static_cast(0x80 + N)); + write_number(static_cast(0x80 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x98)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x99)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x9A)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x9B)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -263,28 +263,28 @@ class binary_writer const auto N = j.m_value.object->size(); if (N <= 0x17) { - write_number(static_cast(0xA0 + N)); + write_number(static_cast(0xA0 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xB8)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xB9)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xBA)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xBB)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -333,31 +333,31 @@ class binary_writer if (j.m_value.number_unsigned < 128) { // positive fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 oa->write_character(to_char_type(0xCC)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 oa->write_character(to_char_type(0xCD)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 oa->write_character(to_char_type(0xCE)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 oa->write_character(to_char_type(0xCF)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } else @@ -365,35 +365,35 @@ class binary_writer if (j.m_value.number_integer >= -32) { // negative fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 8 oa->write_character(to_char_type(0xD0)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 16 oa->write_character(to_char_type(0xD1)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 32 oa->write_character(to_char_type(0xD2)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 64 oa->write_character(to_char_type(0xD3)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } break; @@ -404,31 +404,31 @@ class binary_writer if (j.m_value.number_unsigned < 128) { // positive fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 oa->write_character(to_char_type(0xCC)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 oa->write_character(to_char_type(0xCD)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 oa->write_character(to_char_type(0xCE)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 oa->write_character(to_char_type(0xCF)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } break; } @@ -447,25 +447,25 @@ class binary_writer if (N <= 31) { // fixstr - write_number(static_cast(0xA0 | N)); + write_number(static_cast(0xA0 | N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 8 oa->write_character(to_char_type(0xD9)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 16 oa->write_character(to_char_type(0xDA)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 32 oa->write_character(to_char_type(0xDB)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write the string @@ -482,19 +482,19 @@ class binary_writer if (N <= 15) { // fixarray - write_number(static_cast(0x90 | N)); + write_number(static_cast(0x90 | N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // array 16 oa->write_character(to_char_type(0xDC)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // array 32 oa->write_character(to_char_type(0xDD)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write each element @@ -512,19 +512,19 @@ class binary_writer if (N <= 15) { // fixmap - write_number(static_cast(0x80 | (N & 0xF))); + write_number(static_cast(0x80 | (N & 0xF))); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // map 16 oa->write_character(to_char_type(0xDE)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // map 32 oa->write_character(to_char_type(0xDF)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write each element @@ -790,14 +790,9 @@ class binary_writer */ static std::size_t calc_bson_integer_size(const std::int64_t value) { - if ((std::numeric_limits::min)() <= value and value <= (std::numeric_limits::max)()) - { - return sizeof(std::int32_t); - } - else - { - return sizeof(std::int64_t); - } + return (std::numeric_limits::min)() <= value and value <= (std::numeric_limits::max)() + ? sizeof(std::int32_t) + : sizeof(std::int64_t); } /*! @@ -1065,45 +1060,45 @@ class binary_writer void write_number_with_ubjson_prefix(const NumberType n, const bool add_prefix) { - if (n <= static_cast((std::numeric_limits::max)())) + if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('i')); // int8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= (std::numeric_limits::max)()) + else if (n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('U')); // uint8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('I')); // int16 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('l')); // int32 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('L')); // int64 } - write_number(static_cast(n)); + write_number(static_cast(n)); } else { @@ -1118,45 +1113,45 @@ class binary_writer void write_number_with_ubjson_prefix(const NumberType n, const bool add_prefix) { - if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('i')); // int8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (static_cast((std::numeric_limits::min)()) <= n and n <= static_cast((std::numeric_limits::max)())) + else if (static_cast((std::numeric_limits::min)()) <= n and n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('U')); // uint8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('I')); // int16 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('l')); // int32 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('L')); // int64 } - write_number(static_cast(n)); + write_number(static_cast(n)); } // LCOV_EXCL_START else @@ -1187,19 +1182,19 @@ class binary_writer case value_t::number_integer: { - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'i'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'U'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'I'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'l'; } @@ -1209,19 +1204,19 @@ class binary_writer case value_t::number_unsigned: { - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'i'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'U'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'I'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'l'; } diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index f51afeea..5e85355a 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -45,8 +45,8 @@ class serializer using number_float_t = typename BasicJsonType::number_float_t; using number_integer_t = typename BasicJsonType::number_integer_t; using number_unsigned_t = typename BasicJsonType::number_unsigned_t; - static constexpr uint8_t UTF8_ACCEPT = 0; - static constexpr uint8_t UTF8_REJECT = 1; + static constexpr std::uint8_t UTF8_ACCEPT = 0; + static constexpr std::uint8_t UTF8_REJECT = 1; public: /*! @@ -298,8 +298,8 @@ class serializer */ void dump_escaped(const string_t& s, const bool ensure_ascii) { - uint32_t codepoint; - uint8_t state = UTF8_ACCEPT; + std::uint32_t codepoint; + std::uint8_t state = UTF8_ACCEPT; std::size_t bytes = 0; // number of bytes written to string_buffer // number of bytes written at the point of the last valid byte @@ -374,14 +374,14 @@ class serializer if (codepoint <= 0xFFFF) { (std::snprintf)(string_buffer.data() + bytes, 7, "\\u%04x", - static_cast(codepoint)); + static_cast(codepoint)); bytes += 6; } else { (std::snprintf)(string_buffer.data() + bytes, 13, "\\u%04x\\u%04x", - static_cast(0xD7C0 + (codepoint >> 10)), - static_cast(0xDC00 + (codepoint & 0x3FF))); + static_cast(0xD7C0u + (codepoint >> 10u)), + static_cast(0xDC00u + (codepoint & 0x3FFu))); bytes += 12; } } @@ -508,7 +508,7 @@ class serializer case error_handler_t::strict: { std::string sn(3, '\0'); - (std::snprintf)(&sn[0], sn.size(), "%.2X", static_cast(s.back())); + (std::snprintf)(&sn[0], sn.size(), "%.2X", static_cast(s.back())); JSON_THROW(type_error::create(316, "incomplete UTF-8 string; last byte: 0x" + sn)); } @@ -590,16 +590,16 @@ class serializer static constexpr std::array, 100> digits_to_99 { { - {'0', '0'}, {'0', '1'}, {'0', '2'}, {'0', '3'}, {'0', '4'}, {'0', '5'}, {'0', '6'}, {'0', '7'}, {'0', '8'}, {'0', '9'}, - {'1', '0'}, {'1', '1'}, {'1', '2'}, {'1', '3'}, {'1', '4'}, {'1', '5'}, {'1', '6'}, {'1', '7'}, {'1', '8'}, {'1', '9'}, - {'2', '0'}, {'2', '1'}, {'2', '2'}, {'2', '3'}, {'2', '4'}, {'2', '5'}, {'2', '6'}, {'2', '7'}, {'2', '8'}, {'2', '9'}, - {'3', '0'}, {'3', '1'}, {'3', '2'}, {'3', '3'}, {'3', '4'}, {'3', '5'}, {'3', '6'}, {'3', '7'}, {'3', '8'}, {'3', '9'}, - {'4', '0'}, {'4', '1'}, {'4', '2'}, {'4', '3'}, {'4', '4'}, {'4', '5'}, {'4', '6'}, {'4', '7'}, {'4', '8'}, {'4', '9'}, - {'5', '0'}, {'5', '1'}, {'5', '2'}, {'5', '3'}, {'5', '4'}, {'5', '5'}, {'5', '6'}, {'5', '7'}, {'5', '8'}, {'5', '9'}, - {'6', '0'}, {'6', '1'}, {'6', '2'}, {'6', '3'}, {'6', '4'}, {'6', '5'}, {'6', '6'}, {'6', '7'}, {'6', '8'}, {'6', '9'}, - {'7', '0'}, {'7', '1'}, {'7', '2'}, {'7', '3'}, {'7', '4'}, {'7', '5'}, {'7', '6'}, {'7', '7'}, {'7', '8'}, {'7', '9'}, - {'8', '0'}, {'8', '1'}, {'8', '2'}, {'8', '3'}, {'8', '4'}, {'8', '5'}, {'8', '6'}, {'8', '7'}, {'8', '8'}, {'8', '9'}, - {'9', '0'}, {'9', '1'}, {'9', '2'}, {'9', '3'}, {'9', '4'}, {'9', '5'}, {'9', '6'}, {'9', '7'}, {'9', '8'}, {'9', '9'}, + {{'0', '0'}}, {{'0', '1'}}, {{'0', '2'}}, {{'0', '3'}}, {{'0', '4'}}, {{'0', '5'}}, {{'0', '6'}}, {{'0', '7'}}, {{'0', '8'}}, {{'0', '9'}}, + {{'1', '0'}}, {{'1', '1'}}, {{'1', '2'}}, {{'1', '3'}}, {{'1', '4'}}, {{'1', '5'}}, {{'1', '6'}}, {{'1', '7'}}, {{'1', '8'}}, {{'1', '9'}}, + {{'2', '0'}}, {{'2', '1'}}, {{'2', '2'}}, {{'2', '3'}}, {{'2', '4'}}, {{'2', '5'}}, {{'2', '6'}}, {{'2', '7'}}, {{'2', '8'}}, {{'2', '9'}}, + {{'3', '0'}}, {{'3', '1'}}, {{'3', '2'}}, {{'3', '3'}}, {{'3', '4'}}, {{'3', '5'}}, {{'3', '6'}}, {{'3', '7'}}, {{'3', '8'}}, {{'3', '9'}}, + {{'4', '0'}}, {{'4', '1'}}, {{'4', '2'}}, {{'4', '3'}}, {{'4', '4'}}, {{'4', '5'}}, {{'4', '6'}}, {{'4', '7'}}, {{'4', '8'}}, {{'4', '9'}}, + {{'5', '0'}}, {{'5', '1'}}, {{'5', '2'}}, {{'5', '3'}}, {{'5', '4'}}, {{'5', '5'}}, {{'5', '6'}}, {{'5', '7'}}, {{'5', '8'}}, {{'5', '9'}}, + {{'6', '0'}}, {{'6', '1'}}, {{'6', '2'}}, {{'6', '3'}}, {{'6', '4'}}, {{'6', '5'}}, {{'6', '6'}}, {{'6', '7'}}, {{'6', '8'}}, {{'6', '9'}}, + {{'7', '0'}}, {{'7', '1'}}, {{'7', '2'}}, {{'7', '3'}}, {{'7', '4'}}, {{'7', '5'}}, {{'7', '6'}}, {{'7', '7'}}, {{'7', '8'}}, {{'7', '9'}}, + {{'8', '0'}}, {{'8', '1'}}, {{'8', '2'}}, {{'8', '3'}}, {{'8', '4'}}, {{'8', '5'}}, {{'8', '6'}}, {{'8', '7'}}, {{'8', '8'}}, {{'8', '9'}}, + {{'9', '0'}}, {{'9', '1'}}, {{'9', '2'}}, {{'9', '3'}}, {{'9', '4'}}, {{'9', '5'}}, {{'9', '6'}}, {{'9', '7'}}, {{'9', '8'}}, {{'9', '9'}}, } }; @@ -770,9 +770,9 @@ class serializer @copyright Copyright (c) 2008-2009 Bjoern Hoehrmann @sa http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ */ - static uint8_t decode(uint8_t& state, uint32_t& codep, const uint8_t byte) noexcept + static std::uint8_t decode(std::uint8_t& state, std::uint32_t& codep, const std::uint8_t byte) noexcept { - static const std::array utf8d = + static const std::array utf8d = { { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 00..1F @@ -792,11 +792,11 @@ class serializer } }; - const uint8_t type = utf8d[byte]; + const std::uint8_t type = utf8d[byte]; codep = (state != UTF8_ACCEPT) - ? (byte & 0x3fu) | (codep << 6) - : static_cast(0xff >> type) & (byte); + ? (byte & 0x3fu) | (codep << 6u) + : (0xFFu >> type) & (byte); state = utf8d[256u + state * 16u + type]; return state; diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 60f453d8..3228f9a2 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -6153,13 +6153,9 @@ class basic_json const bool strict = true) { assert(sax); - switch (format) - { - case input_format_t::json: - return parser(std::move(i)).sax_parse(sax, strict); - default: - return detail::binary_reader(std::move(i)).sax_parse(format, sax, strict); - } + return format == input_format_t::json + ? parser(std::move(i)).sax_parse(sax, strict) + : detail::binary_reader(std::move(i)).sax_parse(format, sax, strict); } /*! diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 301f4ed3..897c22d0 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -2170,6 +2170,7 @@ constexpr const auto& to_json = detail::static_const::value; // #include +#include // array #include // assert #include // size_t #include // strlen @@ -2325,7 +2326,11 @@ template struct wide_string_input_helper { // UTF-32 - static void fill_buffer(const WideStringType& str, size_t& current_wchar, std::array::int_type, 4>& utf8_bytes, size_t& utf8_bytes_index, size_t& utf8_bytes_filled) + static void fill_buffer(const WideStringType& str, + size_t& current_wchar, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) { utf8_bytes_index = 0; @@ -2342,34 +2347,34 @@ struct wide_string_input_helper // UTF-32 to UTF-8 encoding if (wc < 0x80) { - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0u | ((wc >> 6u) & 0x1Fu); - utf8_bytes[1] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((wc >> 6u) & 0x1Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 2; } else if (wc <= 0xFFFF) { - utf8_bytes[0] = 0xE0u | ((wc >> 12u) & 0x0Fu); - utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[2] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((wc >> 12u) & 0x0Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 3; } else if (wc <= 0x10FFFF) { - utf8_bytes[0] = 0xF0u | ((wc >> 18u) & 0x07u); - utf8_bytes[1] = 0x80u | ((wc >> 12u) & 0x3Fu); - utf8_bytes[2] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[3] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xF0u | ((wc >> 18u) & 0x07u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 4; } else { // unknown character - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } } @@ -2380,7 +2385,11 @@ template struct wide_string_input_helper { // UTF-16 - static void fill_buffer(const WideStringType& str, size_t& current_wchar, std::array::int_type, 4>& utf8_bytes, size_t& utf8_bytes_index, size_t& utf8_bytes_filled) + static void fill_buffer(const WideStringType& str, + size_t& current_wchar, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) { utf8_bytes_index = 0; @@ -2397,20 +2406,20 @@ struct wide_string_input_helper // UTF-16 to UTF-8 encoding if (wc < 0x80) { - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } else if (wc <= 0x7FF) { - utf8_bytes[0] = 0xC0u | ((wc >> 6u)); - utf8_bytes[1] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((wc >> 6u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 2; } else if (0xD800 > wc or wc >= 0xE000) { - utf8_bytes[0] = 0xE0u | ((wc >> 12u)); - utf8_bytes[1] = 0x80u | ((wc >> 6u) & 0x3Fu); - utf8_bytes[2] = 0x80u | (wc & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((wc >> 12u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((wc >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (wc & 0x3Fu)); utf8_bytes_filled = 3; } else @@ -2419,17 +2428,17 @@ struct wide_string_input_helper { const auto wc2 = static_cast(str[current_wchar++]); const auto charcode = 0x10000u + (((wc & 0x3FFu) << 10u) | (wc2 & 0x3FFu)); - utf8_bytes[0] = 0xF0u | (charcode >> 18u); - utf8_bytes[1] = 0x80u | ((charcode >> 12u) & 0x3Fu); - utf8_bytes[2] = 0x80u | ((charcode >> 6u) & 0x3Fu); - utf8_bytes[3] = 0x80u | (charcode & 0x3Fu); + utf8_bytes[0] = static_cast::int_type>(0xF0u | (charcode >> 18u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((charcode >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((charcode >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (charcode & 0x3Fu)); utf8_bytes_filled = 4; } else { // unknown character ++current_wchar; - utf8_bytes[0] = wc; + utf8_bytes[0] = static_cast::int_type>(wc); utf8_bytes_filled = 1; } } @@ -2596,6 +2605,7 @@ class input_adapter // #include +#include // array #include // localeconv #include // size_t #include // strtof, strtod, strtold, strtoll, strtoull @@ -2755,15 +2765,15 @@ class lexer if (current >= '0' and current <= '9') { - codepoint += ((static_cast(current) - 0x30u) << factor); + codepoint += static_cast((static_cast(current) - 0x30u) << factor); } else if (current >= 'A' and current <= 'F') { - codepoint += ((static_cast(current) - 0x37u) << factor); + codepoint += static_cast((static_cast(current) - 0x37u) << factor); } else if (current >= 'a' and current <= 'f') { - codepoint += ((static_cast(current) - 0x57u) << factor); + codepoint += static_cast((static_cast(current) - 0x57u) << factor); } else { @@ -2921,15 +2931,15 @@ class lexer if (JSON_LIKELY(0xDC00 <= codepoint2 and codepoint2 <= 0xDFFF)) { // overwrite codepoint - codepoint = - // high surrogate occupies the most significant 22 bits - (static_cast(codepoint1) << 10u) - // low surrogate occupies the least significant 15 bits - + static_cast(codepoint2) - // there is still the 0xD800, 0xDC00 and 0x10000 noise - // in the result so we have to subtract with: - // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00u; + codepoint = static_cast( + // high surrogate occupies the most significant 22 bits + (static_cast(codepoint1) << 10u) + // low surrogate occupies the least significant 15 bits + + static_cast(codepoint2) + // there is still the 0xD800, 0xDC00 and 0x10000 noise + // in the result so we have to subtract with: + // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 + - 0x35FDC00u); } else { @@ -2964,23 +2974,23 @@ class lexer else if (codepoint <= 0x7FF) { // 2-byte characters: 110xxxxx 10xxxxxx - add(0xC0u | (static_cast(codepoint) >> 6u)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xC0u | (static_cast(codepoint) >> 6u))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } else if (codepoint <= 0xFFFF) { // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - add(0xE0u | (static_cast(codepoint) >> 12u)); - add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xE0u | (static_cast(codepoint) >> 12u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } else { // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - add(0xF0u | (static_cast(codepoint) >> 18u)); - add(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu)); - add(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu)); - add(0x80u | (static_cast(codepoint) & 0x3Fu)); + add(static_cast(0xF0u | (static_cast(codepoint) >> 18u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); } break; @@ -3959,9 +3969,9 @@ scan_number_done: if ('\x00' <= c and c <= '\x1F') { // escape control characters - char cs[9]; - (std::snprintf)(cs, 9, "", static_cast(c)); - result += cs; + std::array cs{}; + (std::snprintf)(cs.data(), cs.size(), "", static_cast(c)); + result += cs.data(); } else { @@ -6755,9 +6765,9 @@ class binary_reader default: // anything else not supported (yet) { - char cr[3]; - (std::snprintf)(cr, sizeof(cr), "%.2hhX", static_cast(element_type)); - return sax->parse_error(element_type_parse_position, std::string(cr), parse_error::create(114, element_type_parse_position, "Unsupported BSON record type 0x" + std::string(cr))); + std::array cr{}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(element_type)); + return sax->parse_error(element_type_parse_position, std::string(cr.data()), parse_error::create(114, element_type_parse_position, "Unsupported BSON record type 0x" + std::string(cr.data()))); } } } @@ -6880,25 +6890,25 @@ class binary_reader case 0x18: // Unsigned integer (one-byte uint8_t follows) { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x19: // Unsigned integer (two-byte uint16_t follows) { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x1A: // Unsigned integer (four-byte uint32_t follows) { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } case 0x1B: // Unsigned integer (eight-byte uint64_t follows) { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::cbor, number) and sax->number_unsigned(number); } @@ -6927,29 +6937,29 @@ class binary_reader case 0x35: case 0x36: case 0x37: - return sax->number_integer(static_cast(0x20 - 1 - current)); + return sax->number_integer(static_cast(0x20 - 1 - current)); case 0x38: // Negative integer (one-byte uint8_t follows) { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x39: // Negative integer -1-n (two-byte uint16_t follows) { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x3A: // Negative integer -1-n (four-byte uint32_t follows) { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - number); } case 0x3B: // Negative integer -1-n (eight-byte uint64_t follows) { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::cbor, number) and sax->number_integer(static_cast(-1) - static_cast(number)); } @@ -7018,25 +7028,25 @@ class binary_reader case 0x98: // array (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x99: // array (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x9A: // array (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } case 0x9B: // array (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_cbor_array(static_cast(len)); } @@ -7072,25 +7082,25 @@ class binary_reader case 0xB8: // map (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xB9: // map (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xBA: // map (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } case 0xBB: // map (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_cbor_object(static_cast(len)); } @@ -7130,10 +7140,10 @@ class binary_reader // without such support. An example of a small decoder for // half-precision floating-point numbers in the C language // is shown in Fig. 3. - const unsigned int half = (byte1 << 8u) + byte2; + const auto half = static_cast((byte1 << 8u) + byte2); const double val = [&half] { - const unsigned int exp = (half >> 10u) & 0x1Fu; + const int exp = (half >> 10u) & 0x1Fu; const unsigned int mant = half & 0x3FFu; assert(0 <= exp and exp <= 32); assert(0 <= mant and mant <= 1024); @@ -7225,25 +7235,25 @@ class binary_reader case 0x78: // UTF-8 string (one-byte uint8_t for n follows) { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x79: // UTF-8 string (two-byte uint16_t for n follow) { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x7A: // UTF-8 string (four-byte uint32_t for n follow) { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } case 0x7B: // UTF-8 string (eight-byte uint64_t for n follow) { - uint64_t len; + std::uint64_t len; return get_number(input_format_t::cbor, len) and get_string(input_format_t::cbor, len, result); } @@ -7600,49 +7610,49 @@ class binary_reader case 0xCC: // uint 8 { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCD: // uint 16 { - uint16_t number; + std::uint16_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCE: // uint 32 { - uint32_t number; + std::uint32_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xCF: // uint 64 { - uint64_t number; + std::uint64_t number; return get_number(input_format_t::msgpack, number) and sax->number_unsigned(number); } case 0xD0: // int 8 { - int8_t number; + std::int8_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD1: // int 16 { - int16_t number; + std::int16_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD2: // int 32 { - int32_t number; + std::int32_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } case 0xD3: // int 64 { - int64_t number; + std::int64_t number; return get_number(input_format_t::msgpack, number) and sax->number_integer(number); } @@ -7656,25 +7666,25 @@ class binary_reader case 0xDC: // array 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_array(static_cast(len)); } case 0xDD: // array 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_array(static_cast(len)); } case 0xDE: // map 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_object(static_cast(len)); } case 0xDF: // map 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_msgpack_object(static_cast(len)); } @@ -7711,7 +7721,7 @@ class binary_reader case 0xFD: case 0xFE: case 0xFF: - return sax->number_integer(static_cast(current)); + return sax->number_integer(static_cast(current)); default: // anything else { @@ -7779,19 +7789,19 @@ class binary_reader case 0xD9: // str 8 { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } case 0xDA: // str 16 { - uint16_t len; + std::uint16_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } case 0xDB: // str 32 { - uint32_t len; + std::uint32_t len; return get_number(input_format_t::msgpack, len) and get_string(input_format_t::msgpack, len, result); } @@ -7889,7 +7899,7 @@ class binary_reader { if (get_char) { - get(); // TODO: may we ignore N here? + get(); // TODO(niels): may we ignore N here? } if (JSON_UNLIKELY(not unexpect_eof(input_format_t::ubjson, "value"))) @@ -7901,31 +7911,31 @@ class binary_reader { case 'U': { - uint8_t len; + std::uint8_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'i': { - int8_t len; + std::int8_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'I': { - int16_t len; + std::int16_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'l': { - int32_t len; + std::int32_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } case 'L': { - int64_t len; + std::int64_t len; return get_number(input_format_t::ubjson, len) and get_string(input_format_t::ubjson, len, result); } @@ -7945,7 +7955,7 @@ class binary_reader { case 'U': { - uint8_t number; + std::uint8_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -7956,7 +7966,7 @@ class binary_reader case 'i': { - int8_t number; + std::int8_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -7967,7 +7977,7 @@ class binary_reader case 'I': { - int16_t number; + std::int16_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -7978,7 +7988,7 @@ class binary_reader case 'l': { - int32_t number; + std::int32_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -7989,7 +7999,7 @@ class binary_reader case 'L': { - int64_t number; + std::int64_t number; if (JSON_UNLIKELY(not get_number(input_format_t::ubjson, number))) { return false; @@ -8044,10 +8054,12 @@ class binary_reader return get_ubjson_size_value(result.first); } - else if (current == '#') + + if (current == '#') { return get_ubjson_size_value(result.first); } + return true; } @@ -8072,31 +8084,31 @@ class binary_reader case 'U': { - uint8_t number; + std::uint8_t number; return get_number(input_format_t::ubjson, number) and sax->number_unsigned(number); } case 'i': { - int8_t number; + std::int8_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'I': { - int16_t number; + std::int16_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'l': { - int32_t number; + std::int32_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } case 'L': { - int64_t number; + std::int64_t number; return get_number(input_format_t::ubjson, number) and sax->number_integer(number); } @@ -8335,7 +8347,7 @@ class binary_reader bool get_number(const input_format_t format, NumberType& result) { // step 1: read input into array with system's byte order - std::array vec; + std::array vec; for (std::size_t i = 0; i < sizeof(NumberType); ++i) { get(); @@ -8347,11 +8359,11 @@ class binary_reader // reverse byte order prior to conversion if necessary if (is_little_endian != InputIsLittleEndian) { - vec[sizeof(NumberType) - i - 1] = static_cast(current); + vec[sizeof(NumberType) - i - 1] = static_cast(current); } else { - vec[i] = static_cast(current); // LCOV_EXCL_LINE + vec[i] = static_cast(current); // LCOV_EXCL_LINE } } @@ -8412,9 +8424,9 @@ class binary_reader */ std::string get_token_string() const { - char cr[3]; - (std::snprintf)(cr, 3, "%.2hhX", static_cast(current)); - return std::string{cr}; + std::array cr{}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(current)); + return std::string{cr.data()}; } /*! @@ -8568,27 +8580,27 @@ class binary_writer // code from the value_t::number_unsigned case here. if (j.m_value.number_integer <= 0x17) { - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x18)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x19)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x1A)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } else { oa->write_character(to_char_type(0x1B)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } else @@ -8598,27 +8610,27 @@ class binary_writer const auto positive_number = -1 - j.m_value.number_integer; if (j.m_value.number_integer >= -24) { - write_number(static_cast(0x20 + positive_number)); + write_number(static_cast(0x20 + positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x38)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x39)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } - else if (positive_number <= (std::numeric_limits::max)()) + else if (positive_number <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x3A)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } else { oa->write_character(to_char_type(0x3B)); - write_number(static_cast(positive_number)); + write_number(static_cast(positive_number)); } } break; @@ -8628,27 +8640,27 @@ class binary_writer { if (j.m_value.number_unsigned <= 0x17) { - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x18)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x19)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x1A)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } else { oa->write_character(to_char_type(0x1B)); - write_number(static_cast(j.m_value.number_unsigned)); + write_number(static_cast(j.m_value.number_unsigned)); } break; } @@ -8666,28 +8678,28 @@ class binary_writer const auto N = j.m_value.string->size(); if (N <= 0x17) { - write_number(static_cast(0x60 + N)); + write_number(static_cast(0x60 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x78)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x79)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x7A)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x7B)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -8704,28 +8716,28 @@ class binary_writer const auto N = j.m_value.array->size(); if (N <= 0x17) { - write_number(static_cast(0x80 + N)); + write_number(static_cast(0x80 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x98)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x99)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x9A)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0x9B)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -8743,28 +8755,28 @@ class binary_writer const auto N = j.m_value.object->size(); if (N <= 0x17) { - write_number(static_cast(0xA0 + N)); + write_number(static_cast(0xA0 + N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xB8)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xB9)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xBA)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_START - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { oa->write_character(to_char_type(0xBB)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // LCOV_EXCL_STOP @@ -8813,31 +8825,31 @@ class binary_writer if (j.m_value.number_unsigned < 128) { // positive fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 oa->write_character(to_char_type(0xCC)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 oa->write_character(to_char_type(0xCD)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 oa->write_character(to_char_type(0xCE)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 oa->write_character(to_char_type(0xCF)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } else @@ -8845,35 +8857,35 @@ class binary_writer if (j.m_value.number_integer >= -32) { // negative fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 8 oa->write_character(to_char_type(0xD0)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 16 oa->write_character(to_char_type(0xD1)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 32 oa->write_character(to_char_type(0xD2)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_integer >= (std::numeric_limits::min)() and - j.m_value.number_integer <= (std::numeric_limits::max)()) + else if (j.m_value.number_integer >= (std::numeric_limits::min)() and + j.m_value.number_integer <= (std::numeric_limits::max)()) { // int 64 oa->write_character(to_char_type(0xD3)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } } break; @@ -8884,31 +8896,31 @@ class binary_writer if (j.m_value.number_unsigned < 128) { // positive fixnum - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 8 oa->write_character(to_char_type(0xCC)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 16 oa->write_character(to_char_type(0xCD)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 32 oa->write_character(to_char_type(0xCE)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } - else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { // uint 64 oa->write_character(to_char_type(0xCF)); - write_number(static_cast(j.m_value.number_integer)); + write_number(static_cast(j.m_value.number_integer)); } break; } @@ -8927,25 +8939,25 @@ class binary_writer if (N <= 31) { // fixstr - write_number(static_cast(0xA0 | N)); + write_number(static_cast(0xA0 | N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 8 oa->write_character(to_char_type(0xD9)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 16 oa->write_character(to_char_type(0xDA)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // str 32 oa->write_character(to_char_type(0xDB)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write the string @@ -8962,19 +8974,19 @@ class binary_writer if (N <= 15) { // fixarray - write_number(static_cast(0x90 | N)); + write_number(static_cast(0x90 | N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // array 16 oa->write_character(to_char_type(0xDC)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // array 32 oa->write_character(to_char_type(0xDD)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write each element @@ -8992,19 +9004,19 @@ class binary_writer if (N <= 15) { // fixmap - write_number(static_cast(0x80 | (N & 0xF))); + write_number(static_cast(0x80 | (N & 0xF))); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // map 16 oa->write_character(to_char_type(0xDE)); - write_number(static_cast(N)); + write_number(static_cast(N)); } - else if (N <= (std::numeric_limits::max)()) + else if (N <= (std::numeric_limits::max)()) { // map 32 oa->write_character(to_char_type(0xDF)); - write_number(static_cast(N)); + write_number(static_cast(N)); } // step 2: write each element @@ -9270,14 +9282,9 @@ class binary_writer */ static std::size_t calc_bson_integer_size(const std::int64_t value) { - if ((std::numeric_limits::min)() <= value and value <= (std::numeric_limits::max)()) - { - return sizeof(std::int32_t); - } - else - { - return sizeof(std::int64_t); - } + return (std::numeric_limits::min)() <= value and value <= (std::numeric_limits::max)() + ? sizeof(std::int32_t) + : sizeof(std::int64_t); } /*! @@ -9545,45 +9552,45 @@ class binary_writer void write_number_with_ubjson_prefix(const NumberType n, const bool add_prefix) { - if (n <= static_cast((std::numeric_limits::max)())) + if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('i')); // int8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= (std::numeric_limits::max)()) + else if (n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('U')); // uint8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('I')); // int16 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('l')); // int32 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (n <= static_cast((std::numeric_limits::max)())) + else if (n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('L')); // int64 } - write_number(static_cast(n)); + write_number(static_cast(n)); } else { @@ -9598,45 +9605,45 @@ class binary_writer void write_number_with_ubjson_prefix(const NumberType n, const bool add_prefix) { - if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('i')); // int8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if (static_cast((std::numeric_limits::min)()) <= n and n <= static_cast((std::numeric_limits::max)())) + else if (static_cast((std::numeric_limits::min)()) <= n and n <= static_cast((std::numeric_limits::max)())) { if (add_prefix) { oa->write_character(to_char_type('U')); // uint8 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('I')); // int16 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('l')); // int32 } - write_number(static_cast(n)); + write_number(static_cast(n)); } - else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) + else if ((std::numeric_limits::min)() <= n and n <= (std::numeric_limits::max)()) { if (add_prefix) { oa->write_character(to_char_type('L')); // int64 } - write_number(static_cast(n)); + write_number(static_cast(n)); } // LCOV_EXCL_START else @@ -9667,19 +9674,19 @@ class binary_writer case value_t::number_integer: { - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'i'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'U'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'I'; } - if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) + if ((std::numeric_limits::min)() <= j.m_value.number_integer and j.m_value.number_integer <= (std::numeric_limits::max)()) { return 'l'; } @@ -9689,19 +9696,19 @@ class binary_writer case value_t::number_unsigned: { - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'i'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'U'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'I'; } - if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) { return 'l'; } @@ -9841,6 +9848,7 @@ class binary_writer // #include +#include // array #include // assert #include // or, and, not #include // signbit, isfinite @@ -9890,10 +9898,10 @@ struct diyfp // f * 2^e { static constexpr int kPrecision = 64; // = q - uint64_t f = 0; + std::uint64_t f = 0; int e = 0; - constexpr diyfp(uint64_t f_, int e_) noexcept : f(f_), e(e_) {} + constexpr diyfp(std::uint64_t f_, int e_) noexcept : f(f_), e(e_) {} /*! @brief returns x - y @@ -9938,23 +9946,23 @@ struct diyfp // f * 2^e // // = p_lo + 2^64 p_hi - const uint64_t u_lo = x.f & 0xFFFFFFFF; - const uint64_t u_hi = x.f >> 32; - const uint64_t v_lo = y.f & 0xFFFFFFFF; - const uint64_t v_hi = y.f >> 32; + const std::uint64_t u_lo = x.f & 0xFFFFFFFFu; + const std::uint64_t u_hi = x.f >> 32u; + const std::uint64_t v_lo = y.f & 0xFFFFFFFFu; + const std::uint64_t v_hi = y.f >> 32u; - const uint64_t p0 = u_lo * v_lo; - const uint64_t p1 = u_lo * v_hi; - const uint64_t p2 = u_hi * v_lo; - const uint64_t p3 = u_hi * v_hi; + const std::uint64_t p0 = u_lo * v_lo; + const std::uint64_t p1 = u_lo * v_hi; + const std::uint64_t p2 = u_hi * v_lo; + const std::uint64_t p3 = u_hi * v_hi; - const uint64_t p0_hi = p0 >> 32; - const uint64_t p1_lo = p1 & 0xFFFFFFFF; - const uint64_t p1_hi = p1 >> 32; - const uint64_t p2_lo = p2 & 0xFFFFFFFF; - const uint64_t p2_hi = p2 >> 32; + const std::uint64_t p0_hi = p0 >> 32u; + const std::uint64_t p1_lo = p1 & 0xFFFFFFFFu; + const std::uint64_t p1_hi = p1 >> 32u; + const std::uint64_t p2_lo = p2 & 0xFFFFFFFFu; + const std::uint64_t p2_hi = p2 >> 32u; - uint64_t Q = p0_hi + p1_lo + p2_lo; + std::uint64_t Q = p0_hi + p1_lo + p2_lo; // The full product might now be computed as // @@ -9965,9 +9973,9 @@ struct diyfp // f * 2^e // Effectively we only need to add the highest bit in p_lo to p_hi (and // Q_hi + 1 does not overflow). - Q += uint64_t{1} << (64 - 32 - 1); // round, ties up + Q += std::uint64_t{1} << (64u - 32u - 1u); // round, ties up - const uint64_t h = p3 + p2_hi + p1_hi + (Q >> 32); + const std::uint64_t h = p3 + p2_hi + p1_hi + (Q >> 32u); return {h, x.e + y.e + 64}; } @@ -9980,9 +9988,9 @@ struct diyfp // f * 2^e { assert(x.f != 0); - while ((x.f >> 63) == 0) + while ((x.f >> 63u) == 0) { - x.f <<= 1; + x.f <<= 1u; x.e--; } @@ -10036,13 +10044,13 @@ boundaries compute_boundaries(FloatType value) constexpr int kPrecision = std::numeric_limits::digits; // = p (includes the hidden bit) constexpr int kBias = std::numeric_limits::max_exponent - 1 + (kPrecision - 1); constexpr int kMinExp = 1 - kBias; - constexpr uint64_t kHiddenBit = uint64_t{1} << (kPrecision - 1); // = 2^(p-1) + constexpr std::uint64_t kHiddenBit = std::uint64_t{1} << (kPrecision - 1); // = 2^(p-1) - using bits_type = typename std::conditional< kPrecision == 24, uint32_t, uint64_t >::type; + using bits_type = typename std::conditional::type; - const uint64_t bits = reinterpret_bits(value); - const uint64_t E = bits >> (kPrecision - 1); - const uint64_t F = bits & (kHiddenBit - 1); + const std::uint64_t bits = reinterpret_bits(value); + const std::uint64_t E = bits >> (kPrecision - 1); + const std::uint64_t F = bits & (kHiddenBit - 1); const bool is_denormal = (E == 0); const diyfp v = is_denormal @@ -10145,7 +10153,7 @@ constexpr int kGamma = -32; struct cached_power // c = f * 2^e ~= 10^k { - uint64_t f; + std::uint64_t f; int e; int k; }; @@ -10209,91 +10217,92 @@ inline cached_power get_cached_power_for_binary_exponent(int e) // NB: // Actually this function returns c, such that -60 <= e_c + e + 64 <= -34. - constexpr int kCachedPowersSize = 79; constexpr int kCachedPowersMinDecExp = -300; constexpr int kCachedPowersDecStep = 8; - static constexpr cached_power kCachedPowers[] = + static constexpr std::array kCachedPowers = { - { 0xAB70FE17C79AC6CA, -1060, -300 }, - { 0xFF77B1FCBEBCDC4F, -1034, -292 }, - { 0xBE5691EF416BD60C, -1007, -284 }, - { 0x8DD01FAD907FFC3C, -980, -276 }, - { 0xD3515C2831559A83, -954, -268 }, - { 0x9D71AC8FADA6C9B5, -927, -260 }, - { 0xEA9C227723EE8BCB, -901, -252 }, - { 0xAECC49914078536D, -874, -244 }, - { 0x823C12795DB6CE57, -847, -236 }, - { 0xC21094364DFB5637, -821, -228 }, - { 0x9096EA6F3848984F, -794, -220 }, - { 0xD77485CB25823AC7, -768, -212 }, - { 0xA086CFCD97BF97F4, -741, -204 }, - { 0xEF340A98172AACE5, -715, -196 }, - { 0xB23867FB2A35B28E, -688, -188 }, - { 0x84C8D4DFD2C63F3B, -661, -180 }, - { 0xC5DD44271AD3CDBA, -635, -172 }, - { 0x936B9FCEBB25C996, -608, -164 }, - { 0xDBAC6C247D62A584, -582, -156 }, - { 0xA3AB66580D5FDAF6, -555, -148 }, - { 0xF3E2F893DEC3F126, -529, -140 }, - { 0xB5B5ADA8AAFF80B8, -502, -132 }, - { 0x87625F056C7C4A8B, -475, -124 }, - { 0xC9BCFF6034C13053, -449, -116 }, - { 0x964E858C91BA2655, -422, -108 }, - { 0xDFF9772470297EBD, -396, -100 }, - { 0xA6DFBD9FB8E5B88F, -369, -92 }, - { 0xF8A95FCF88747D94, -343, -84 }, - { 0xB94470938FA89BCF, -316, -76 }, - { 0x8A08F0F8BF0F156B, -289, -68 }, - { 0xCDB02555653131B6, -263, -60 }, - { 0x993FE2C6D07B7FAC, -236, -52 }, - { 0xE45C10C42A2B3B06, -210, -44 }, - { 0xAA242499697392D3, -183, -36 }, - { 0xFD87B5F28300CA0E, -157, -28 }, - { 0xBCE5086492111AEB, -130, -20 }, - { 0x8CBCCC096F5088CC, -103, -12 }, - { 0xD1B71758E219652C, -77, -4 }, - { 0x9C40000000000000, -50, 4 }, - { 0xE8D4A51000000000, -24, 12 }, - { 0xAD78EBC5AC620000, 3, 20 }, - { 0x813F3978F8940984, 30, 28 }, - { 0xC097CE7BC90715B3, 56, 36 }, - { 0x8F7E32CE7BEA5C70, 83, 44 }, - { 0xD5D238A4ABE98068, 109, 52 }, - { 0x9F4F2726179A2245, 136, 60 }, - { 0xED63A231D4C4FB27, 162, 68 }, - { 0xB0DE65388CC8ADA8, 189, 76 }, - { 0x83C7088E1AAB65DB, 216, 84 }, - { 0xC45D1DF942711D9A, 242, 92 }, - { 0x924D692CA61BE758, 269, 100 }, - { 0xDA01EE641A708DEA, 295, 108 }, - { 0xA26DA3999AEF774A, 322, 116 }, - { 0xF209787BB47D6B85, 348, 124 }, - { 0xB454E4A179DD1877, 375, 132 }, - { 0x865B86925B9BC5C2, 402, 140 }, - { 0xC83553C5C8965D3D, 428, 148 }, - { 0x952AB45CFA97A0B3, 455, 156 }, - { 0xDE469FBD99A05FE3, 481, 164 }, - { 0xA59BC234DB398C25, 508, 172 }, - { 0xF6C69A72A3989F5C, 534, 180 }, - { 0xB7DCBF5354E9BECE, 561, 188 }, - { 0x88FCF317F22241E2, 588, 196 }, - { 0xCC20CE9BD35C78A5, 614, 204 }, - { 0x98165AF37B2153DF, 641, 212 }, - { 0xE2A0B5DC971F303A, 667, 220 }, - { 0xA8D9D1535CE3B396, 694, 228 }, - { 0xFB9B7CD9A4A7443C, 720, 236 }, - { 0xBB764C4CA7A44410, 747, 244 }, - { 0x8BAB8EEFB6409C1A, 774, 252 }, - { 0xD01FEF10A657842C, 800, 260 }, - { 0x9B10A4E5E9913129, 827, 268 }, - { 0xE7109BFBA19C0C9D, 853, 276 }, - { 0xAC2820D9623BF429, 880, 284 }, - { 0x80444B5E7AA7CF85, 907, 292 }, - { 0xBF21E44003ACDD2D, 933, 300 }, - { 0x8E679C2F5E44FF8F, 960, 308 }, - { 0xD433179D9C8CB841, 986, 316 }, - { 0x9E19DB92B4E31BA9, 1013, 324 }, + { + { 0xAB70FE17C79AC6CA, -1060, -300 }, + { 0xFF77B1FCBEBCDC4F, -1034, -292 }, + { 0xBE5691EF416BD60C, -1007, -284 }, + { 0x8DD01FAD907FFC3C, -980, -276 }, + { 0xD3515C2831559A83, -954, -268 }, + { 0x9D71AC8FADA6C9B5, -927, -260 }, + { 0xEA9C227723EE8BCB, -901, -252 }, + { 0xAECC49914078536D, -874, -244 }, + { 0x823C12795DB6CE57, -847, -236 }, + { 0xC21094364DFB5637, -821, -228 }, + { 0x9096EA6F3848984F, -794, -220 }, + { 0xD77485CB25823AC7, -768, -212 }, + { 0xA086CFCD97BF97F4, -741, -204 }, + { 0xEF340A98172AACE5, -715, -196 }, + { 0xB23867FB2A35B28E, -688, -188 }, + { 0x84C8D4DFD2C63F3B, -661, -180 }, + { 0xC5DD44271AD3CDBA, -635, -172 }, + { 0x936B9FCEBB25C996, -608, -164 }, + { 0xDBAC6C247D62A584, -582, -156 }, + { 0xA3AB66580D5FDAF6, -555, -148 }, + { 0xF3E2F893DEC3F126, -529, -140 }, + { 0xB5B5ADA8AAFF80B8, -502, -132 }, + { 0x87625F056C7C4A8B, -475, -124 }, + { 0xC9BCFF6034C13053, -449, -116 }, + { 0x964E858C91BA2655, -422, -108 }, + { 0xDFF9772470297EBD, -396, -100 }, + { 0xA6DFBD9FB8E5B88F, -369, -92 }, + { 0xF8A95FCF88747D94, -343, -84 }, + { 0xB94470938FA89BCF, -316, -76 }, + { 0x8A08F0F8BF0F156B, -289, -68 }, + { 0xCDB02555653131B6, -263, -60 }, + { 0x993FE2C6D07B7FAC, -236, -52 }, + { 0xE45C10C42A2B3B06, -210, -44 }, + { 0xAA242499697392D3, -183, -36 }, + { 0xFD87B5F28300CA0E, -157, -28 }, + { 0xBCE5086492111AEB, -130, -20 }, + { 0x8CBCCC096F5088CC, -103, -12 }, + { 0xD1B71758E219652C, -77, -4 }, + { 0x9C40000000000000, -50, 4 }, + { 0xE8D4A51000000000, -24, 12 }, + { 0xAD78EBC5AC620000, 3, 20 }, + { 0x813F3978F8940984, 30, 28 }, + { 0xC097CE7BC90715B3, 56, 36 }, + { 0x8F7E32CE7BEA5C70, 83, 44 }, + { 0xD5D238A4ABE98068, 109, 52 }, + { 0x9F4F2726179A2245, 136, 60 }, + { 0xED63A231D4C4FB27, 162, 68 }, + { 0xB0DE65388CC8ADA8, 189, 76 }, + { 0x83C7088E1AAB65DB, 216, 84 }, + { 0xC45D1DF942711D9A, 242, 92 }, + { 0x924D692CA61BE758, 269, 100 }, + { 0xDA01EE641A708DEA, 295, 108 }, + { 0xA26DA3999AEF774A, 322, 116 }, + { 0xF209787BB47D6B85, 348, 124 }, + { 0xB454E4A179DD1877, 375, 132 }, + { 0x865B86925B9BC5C2, 402, 140 }, + { 0xC83553C5C8965D3D, 428, 148 }, + { 0x952AB45CFA97A0B3, 455, 156 }, + { 0xDE469FBD99A05FE3, 481, 164 }, + { 0xA59BC234DB398C25, 508, 172 }, + { 0xF6C69A72A3989F5C, 534, 180 }, + { 0xB7DCBF5354E9BECE, 561, 188 }, + { 0x88FCF317F22241E2, 588, 196 }, + { 0xCC20CE9BD35C78A5, 614, 204 }, + { 0x98165AF37B2153DF, 641, 212 }, + { 0xE2A0B5DC971F303A, 667, 220 }, + { 0xA8D9D1535CE3B396, 694, 228 }, + { 0xFB9B7CD9A4A7443C, 720, 236 }, + { 0xBB764C4CA7A44410, 747, 244 }, + { 0x8BAB8EEFB6409C1A, 774, 252 }, + { 0xD01FEF10A657842C, 800, 260 }, + { 0x9B10A4E5E9913129, 827, 268 }, + { 0xE7109BFBA19C0C9D, 853, 276 }, + { 0xAC2820D9623BF429, 880, 284 }, + { 0x80444B5E7AA7CF85, 907, 292 }, + { 0xBF21E44003ACDD2D, 933, 300 }, + { 0x8E679C2F5E44FF8F, 960, 308 }, + { 0xD433179D9C8CB841, 986, 316 }, + { 0x9E19DB92B4E31BA9, 1013, 324 }, + } }; // This computation gives exactly the same results for k as @@ -10307,10 +10316,9 @@ inline cached_power get_cached_power_for_binary_exponent(int e) const int index = (-kCachedPowersMinDecExp + k + (kCachedPowersDecStep - 1)) / kCachedPowersDecStep; assert(index >= 0); - assert(index < kCachedPowersSize); - static_cast(kCachedPowersSize); // Fix warning. + assert(static_cast(index) < kCachedPowers.size()); - const cached_power cached = kCachedPowers[index]; + const cached_power cached = kCachedPowers[static_cast(index)]; assert(kAlpha <= cached.e + e + 64); assert(kGamma >= cached.e + e + 64); @@ -10321,7 +10329,7 @@ inline cached_power get_cached_power_for_binary_exponent(int e) For n != 0, returns k, such that pow10 := 10^(k-1) <= n < 10^k. For n == 0, returns 1 and sets pow10 := 1. */ -inline int find_largest_pow10(const uint32_t n, uint32_t& pow10) +inline int find_largest_pow10(const std::uint32_t n, std::uint32_t& pow10) { // LCOV_EXCL_START if (n >= 1000000000) @@ -10377,8 +10385,8 @@ inline int find_largest_pow10(const uint32_t n, uint32_t& pow10) } } -inline void grisu2_round(char* buf, int len, uint64_t dist, uint64_t delta, - uint64_t rest, uint64_t ten_k) +inline void grisu2_round(char* buf, int len, std::uint64_t dist, std::uint64_t delta, + std::uint64_t rest, std::uint64_t ten_k) { assert(len >= 1); assert(dist <= delta); @@ -10439,8 +10447,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, assert(M_plus.e >= kAlpha); assert(M_plus.e <= kGamma); - uint64_t delta = diyfp::sub(M_plus, M_minus).f; // (significand of (M+ - M-), implicit exponent is e) - uint64_t dist = diyfp::sub(M_plus, w ).f; // (significand of (M+ - w ), implicit exponent is e) + std::uint64_t delta = diyfp::sub(M_plus, M_minus).f; // (significand of (M+ - M-), implicit exponent is e) + std::uint64_t dist = diyfp::sub(M_plus, w ).f; // (significand of (M+ - w ), implicit exponent is e) // Split M+ = f * 2^e into two parts p1 and p2 (note: e < 0): // @@ -10449,10 +10457,10 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // = ((p1 ) * 2^-e + (p2 )) * 2^e // = p1 + p2 * 2^e - const diyfp one(uint64_t{1} << -M_plus.e, M_plus.e); + const diyfp one(std::uint64_t{1} << -M_plus.e, M_plus.e); - auto p1 = static_cast(M_plus.f >> -one.e); // p1 = f div 2^-e (Since -e >= 32, p1 fits into a 32-bit int.) - uint64_t p2 = M_plus.f & (one.f - 1); // p2 = f mod 2^-e + auto p1 = static_cast(M_plus.f >> -one.e); // p1 = f div 2^-e (Since -e >= 32, p1 fits into a 32-bit int.) + std::uint64_t p2 = M_plus.f & (one.f - 1); // p2 = f mod 2^-e // 1) // @@ -10460,7 +10468,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, assert(p1 > 0); - uint32_t pow10; + std::uint32_t pow10; const int k = find_largest_pow10(p1, pow10); // 10^(k-1) <= p1 < 10^k, pow10 = 10^(k-1) @@ -10488,8 +10496,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // M+ = buffer * 10^n + (p1 + p2 * 2^e) (buffer = 0 for n = k) // pow10 = 10^(n-1) <= p1 < 10^n // - const uint32_t d = p1 / pow10; // d = p1 div 10^(n-1) - const uint32_t r = p1 % pow10; // r = p1 mod 10^(n-1) + const std::uint32_t d = p1 / pow10; // d = p1 div 10^(n-1) + const std::uint32_t r = p1 % pow10; // r = p1 mod 10^(n-1) // // M+ = buffer * 10^n + (d * 10^(n-1) + r) + p2 * 2^e // = (buffer * 10 + d) * 10^(n-1) + (r + p2 * 2^e) @@ -10514,7 +10522,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // Note: // Since rest and delta share the same exponent e, it suffices to // compare the significands. - const uint64_t rest = (uint64_t{p1} << -one.e) + p2; + const std::uint64_t rest = (std::uint64_t{p1} << -one.e) + p2; if (rest <= delta) { // V = buffer * 10^n, with M- <= V <= M+. @@ -10530,7 +10538,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // // 10^n = (10^n * 2^-e) * 2^e = ulp * 2^e // - const uint64_t ten_n = uint64_t{pow10} << -one.e; + const std::uint64_t ten_n = std::uint64_t{pow10} << -one.e; grisu2_round(buffer, length, dist, delta, rest, ten_n); return; @@ -10594,8 +10602,8 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // assert(p2 <= std::numeric_limits::max() / 10); p2 *= 10; - const uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e - const uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e + const std::uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e + const std::uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e // // M+ = buffer * 10^-m + 10^-m * (1/10 * (d * 2^-e + r) * 2^e // = buffer * 10^-m + 10^-m * (1/10 * (d + r * 2^e)) @@ -10635,7 +10643,7 @@ inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, // // 10^m * 10^-m = 1 = 2^-e * 2^e = ten_m * 2^e // - const uint64_t ten_m = one.f; + const std::uint64_t ten_m = one.f; grisu2_round(buffer, length, dist, delta, p2, ten_m); // By construction this algorithm generates the shortest possible decimal @@ -10770,7 +10778,7 @@ inline char* append_exponent(char* buf, int e) *buf++ = '+'; } - auto k = static_cast(e); + auto k = static_cast(e); if (k < 10) { // Always print at least two digits in the exponent. @@ -10970,8 +10978,8 @@ class serializer using number_float_t = typename BasicJsonType::number_float_t; using number_integer_t = typename BasicJsonType::number_integer_t; using number_unsigned_t = typename BasicJsonType::number_unsigned_t; - static constexpr uint8_t UTF8_ACCEPT = 0; - static constexpr uint8_t UTF8_REJECT = 1; + static constexpr std::uint8_t UTF8_ACCEPT = 0; + static constexpr std::uint8_t UTF8_REJECT = 1; public: /*! @@ -11223,8 +11231,8 @@ class serializer */ void dump_escaped(const string_t& s, const bool ensure_ascii) { - uint32_t codepoint; - uint8_t state = UTF8_ACCEPT; + std::uint32_t codepoint; + std::uint8_t state = UTF8_ACCEPT; std::size_t bytes = 0; // number of bytes written to string_buffer // number of bytes written at the point of the last valid byte @@ -11299,14 +11307,14 @@ class serializer if (codepoint <= 0xFFFF) { (std::snprintf)(string_buffer.data() + bytes, 7, "\\u%04x", - static_cast(codepoint)); + static_cast(codepoint)); bytes += 6; } else { (std::snprintf)(string_buffer.data() + bytes, 13, "\\u%04x\\u%04x", - static_cast(0xD7C0 + (codepoint >> 10)), - static_cast(0xDC00 + (codepoint & 0x3FF))); + static_cast(0xD7C0u + (codepoint >> 10u)), + static_cast(0xDC00u + (codepoint & 0x3FFu))); bytes += 12; } } @@ -11433,7 +11441,7 @@ class serializer case error_handler_t::strict: { std::string sn(3, '\0'); - (std::snprintf)(&sn[0], sn.size(), "%.2X", static_cast(s.back())); + (std::snprintf)(&sn[0], sn.size(), "%.2X", static_cast(s.back())); JSON_THROW(type_error::create(316, "incomplete UTF-8 string; last byte: 0x" + sn)); } @@ -11515,16 +11523,16 @@ class serializer static constexpr std::array, 100> digits_to_99 { { - {'0', '0'}, {'0', '1'}, {'0', '2'}, {'0', '3'}, {'0', '4'}, {'0', '5'}, {'0', '6'}, {'0', '7'}, {'0', '8'}, {'0', '9'}, - {'1', '0'}, {'1', '1'}, {'1', '2'}, {'1', '3'}, {'1', '4'}, {'1', '5'}, {'1', '6'}, {'1', '7'}, {'1', '8'}, {'1', '9'}, - {'2', '0'}, {'2', '1'}, {'2', '2'}, {'2', '3'}, {'2', '4'}, {'2', '5'}, {'2', '6'}, {'2', '7'}, {'2', '8'}, {'2', '9'}, - {'3', '0'}, {'3', '1'}, {'3', '2'}, {'3', '3'}, {'3', '4'}, {'3', '5'}, {'3', '6'}, {'3', '7'}, {'3', '8'}, {'3', '9'}, - {'4', '0'}, {'4', '1'}, {'4', '2'}, {'4', '3'}, {'4', '4'}, {'4', '5'}, {'4', '6'}, {'4', '7'}, {'4', '8'}, {'4', '9'}, - {'5', '0'}, {'5', '1'}, {'5', '2'}, {'5', '3'}, {'5', '4'}, {'5', '5'}, {'5', '6'}, {'5', '7'}, {'5', '8'}, {'5', '9'}, - {'6', '0'}, {'6', '1'}, {'6', '2'}, {'6', '3'}, {'6', '4'}, {'6', '5'}, {'6', '6'}, {'6', '7'}, {'6', '8'}, {'6', '9'}, - {'7', '0'}, {'7', '1'}, {'7', '2'}, {'7', '3'}, {'7', '4'}, {'7', '5'}, {'7', '6'}, {'7', '7'}, {'7', '8'}, {'7', '9'}, - {'8', '0'}, {'8', '1'}, {'8', '2'}, {'8', '3'}, {'8', '4'}, {'8', '5'}, {'8', '6'}, {'8', '7'}, {'8', '8'}, {'8', '9'}, - {'9', '0'}, {'9', '1'}, {'9', '2'}, {'9', '3'}, {'9', '4'}, {'9', '5'}, {'9', '6'}, {'9', '7'}, {'9', '8'}, {'9', '9'}, + {{'0', '0'}}, {{'0', '1'}}, {{'0', '2'}}, {{'0', '3'}}, {{'0', '4'}}, {{'0', '5'}}, {{'0', '6'}}, {{'0', '7'}}, {{'0', '8'}}, {{'0', '9'}}, + {{'1', '0'}}, {{'1', '1'}}, {{'1', '2'}}, {{'1', '3'}}, {{'1', '4'}}, {{'1', '5'}}, {{'1', '6'}}, {{'1', '7'}}, {{'1', '8'}}, {{'1', '9'}}, + {{'2', '0'}}, {{'2', '1'}}, {{'2', '2'}}, {{'2', '3'}}, {{'2', '4'}}, {{'2', '5'}}, {{'2', '6'}}, {{'2', '7'}}, {{'2', '8'}}, {{'2', '9'}}, + {{'3', '0'}}, {{'3', '1'}}, {{'3', '2'}}, {{'3', '3'}}, {{'3', '4'}}, {{'3', '5'}}, {{'3', '6'}}, {{'3', '7'}}, {{'3', '8'}}, {{'3', '9'}}, + {{'4', '0'}}, {{'4', '1'}}, {{'4', '2'}}, {{'4', '3'}}, {{'4', '4'}}, {{'4', '5'}}, {{'4', '6'}}, {{'4', '7'}}, {{'4', '8'}}, {{'4', '9'}}, + {{'5', '0'}}, {{'5', '1'}}, {{'5', '2'}}, {{'5', '3'}}, {{'5', '4'}}, {{'5', '5'}}, {{'5', '6'}}, {{'5', '7'}}, {{'5', '8'}}, {{'5', '9'}}, + {{'6', '0'}}, {{'6', '1'}}, {{'6', '2'}}, {{'6', '3'}}, {{'6', '4'}}, {{'6', '5'}}, {{'6', '6'}}, {{'6', '7'}}, {{'6', '8'}}, {{'6', '9'}}, + {{'7', '0'}}, {{'7', '1'}}, {{'7', '2'}}, {{'7', '3'}}, {{'7', '4'}}, {{'7', '5'}}, {{'7', '6'}}, {{'7', '7'}}, {{'7', '8'}}, {{'7', '9'}}, + {{'8', '0'}}, {{'8', '1'}}, {{'8', '2'}}, {{'8', '3'}}, {{'8', '4'}}, {{'8', '5'}}, {{'8', '6'}}, {{'8', '7'}}, {{'8', '8'}}, {{'8', '9'}}, + {{'9', '0'}}, {{'9', '1'}}, {{'9', '2'}}, {{'9', '3'}}, {{'9', '4'}}, {{'9', '5'}}, {{'9', '6'}}, {{'9', '7'}}, {{'9', '8'}}, {{'9', '9'}}, } }; @@ -11695,9 +11703,9 @@ class serializer @copyright Copyright (c) 2008-2009 Bjoern Hoehrmann @sa http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ */ - static uint8_t decode(uint8_t& state, uint32_t& codep, const uint8_t byte) noexcept + static std::uint8_t decode(std::uint8_t& state, std::uint32_t& codep, const std::uint8_t byte) noexcept { - static const std::array utf8d = + static const std::array utf8d = { { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 00..1F @@ -11717,11 +11725,11 @@ class serializer } }; - const uint8_t type = utf8d[byte]; + const std::uint8_t type = utf8d[byte]; codep = (state != UTF8_ACCEPT) - ? (byte & 0x3fu) | (codep << 6) - : static_cast(0xff >> type) & (byte); + ? (byte & 0x3fu) | (codep << 6u) + : (0xFFu >> type) & (byte); state = utf8d[256u + state * 16u + type]; return state; @@ -18740,13 +18748,9 @@ class basic_json const bool strict = true) { assert(sax); - switch (format) - { - case input_format_t::json: - return parser(std::move(i)).sax_parse(sax, strict); - default: - return detail::binary_reader(std::move(i)).sax_parse(format, sax, strict); - } + return format == input_format_t::json + ? parser(std::move(i)).sax_parse(sax, strict) + : detail::binary_reader(std::move(i)).sax_parse(format, sax, strict); } /*! From baf8b4be7cb95c8054191a595b3ad20a0a095dd5 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 00:28:14 +0100 Subject: [PATCH 035/220] :heavy_plus_sign: adding cpplint --- third_party/cpplint/LICENSE | 27 + third_party/cpplint/README.rst | 80 + third_party/cpplint/cpplint.py | 6583 ++++++++++++++++++++++++++++++++ 3 files changed, 6690 insertions(+) create mode 100755 third_party/cpplint/LICENSE create mode 100755 third_party/cpplint/README.rst create mode 100755 third_party/cpplint/cpplint.py diff --git a/third_party/cpplint/LICENSE b/third_party/cpplint/LICENSE new file mode 100755 index 00000000..1756a599 --- /dev/null +++ b/third_party/cpplint/LICENSE @@ -0,0 +1,27 @@ +cpplint.py and its corresponding unit tests are Copyright (C) 2009 Google Inc. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/third_party/cpplint/README.rst b/third_party/cpplint/README.rst new file mode 100755 index 00000000..4208f35e --- /dev/null +++ b/third_party/cpplint/README.rst @@ -0,0 +1,80 @@ +cpplint - static code checker for C++ +===================================== + +.. image:: https://travis-ci.org/cpplint/cpplint.svg?branch=master + :target: https://travis-ci.org/cpplint/cpplint + +.. image:: https://img.shields.io/pypi/v/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/pyversions/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/status/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/l/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/dd/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/dw/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +.. image:: https://img.shields.io/pypi/dm/cpplint.svg + :target: https://pypi.python.org/pypi/cpplint + +Cpplint is a command-line tool to check C/C++ files for style issues following `Google's C++ style guide `_. +Cpplint is developed and maintained by Google Inc. at `google/styleguide `_, also see see the `wikipedia entry `_ + +While Google maintains cpplint, Google is not (very) responsive to issues and pull requests, this fork aims to be (somewhat) more open to add fixes to cpplint to enable fixes, when those fixes make cpplint usable in wider contexts. + + +Installation +============ + + +To install cpplint from PyPI, run: + +.. code-block:: bash + + $ pip install cpplint + +Then run it with: + +.. code-block:: bash + + $ cpplint [OPTIONS] files + +For full usage instructions, run: + +.. code-block:: bash + + $ cpplint --help + +Changes +------- + +The modifications in this fork are minor fixes and cosmetic changes: + +* more default extensions +* python 3 compatibility +* customizable file extensions with the --extensions argument +* continuous integration on travis +* support for recursive file discovery via the --recursive argument +* support for excluding files via --exclude +* JUnit XML output format +* Overriding repository root auto-detection via --repository +* Support ``#pragma once`` as an alternative to header include guards + + +Acknowledgements +---------------- + +Thanks to Google Inc. for open-sourcing their in-house tool. +Thanks to maintainers of the fork + +* `tkruse `_ +* `mattyclarkson `_ +* `theandrewdavis `_ diff --git a/third_party/cpplint/cpplint.py b/third_party/cpplint/cpplint.py new file mode 100755 index 00000000..81bc98b7 --- /dev/null +++ b/third_party/cpplint/cpplint.py @@ -0,0 +1,6583 @@ +#!/usr/bin/env python +# +# Copyright (c) 2009 Google Inc. All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +# * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +# * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +# * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +"""Does google-lint on c++ files. + +The goal of this script is to identify places in the code that *may* +be in non-compliance with google style. It does not attempt to fix +up these problems -- the point is to educate. It does also not +attempt to find all problems, or to ensure that everything it does +find is legitimately a problem. + +In particular, we can get very confused by /* and // inside strings! +We do a small hack, which is to ignore //'s with "'s after them on the +same line, but it is far from perfect (in either direction). +""" + +import codecs +import copy +import getopt +import glob +import itertools +import math # for log +import os +import re +import sre_compile +import string +import sys +import sysconfig +import unicodedata +import xml.etree.ElementTree + +# if empty, use defaults +_valid_extensions = set([]) + +__VERSION__ = '1.4.4' + +try: + xrange # Python 2 +except NameError: + # -- pylint: disable=redefined-builtin + xrange = range # Python 3 + + +_USAGE = """ +Syntax: cpplint.py [--verbose=#] [--output=emacs|eclipse|vs7|junit] + [--filter=-x,+y,...] + [--counting=total|toplevel|detailed] [--root=subdir] + [--repository=path] + [--linelength=digits] [--headers=x,y,...] + [--recursive] + [--exclude=path] + [--extensions=hpp,cpp,...] + [--quiet] + [--version] + [file] ... + + Style checker for C/C++ source files. + This is a fork of the Google style checker with minor extensions. + + The style guidelines this tries to follow are those in + https://google.github.io/styleguide/cppguide.html + + Every problem is given a confidence score from 1-5, with 5 meaning we are + certain of the problem, and 1 meaning it could be a legitimate construct. + This will miss some errors, and is not a substitute for a code review. + + To suppress false-positive errors of a certain category, add a + 'NOLINT(category)' comment to the line. NOLINT or NOLINT(*) + suppresses errors of all categories on that line. + + The files passed in will be linted; at least one file must be provided. + Default linted extensions are %s. + Other file types will be ignored. + Change the extensions with the --extensions flag. + + Flags: + + output=emacs|eclipse|vs7|junit + By default, the output is formatted to ease emacs parsing. Visual Studio + compatible output (vs7) may also be used. Further support exists for + eclipse (eclipse), and JUnit (junit). XML parsers such as those used + in Jenkins and Bamboo may also be used. Other formats are unsupported. + + verbose=# + Specify a number 0-5 to restrict errors to certain verbosity levels. + Errors with lower verbosity levels have lower confidence and are more + likely to be false positives. + + quiet + Don't print anything if no errors are found. + + filter=-x,+y,... + Specify a comma-separated list of category-filters to apply: only + error messages whose category names pass the filters will be printed. + (Category names are printed with the message and look like + "[whitespace/indent]".) Filters are evaluated left to right. + "-FOO" and "FOO" means "do not print categories that start with FOO". + "+FOO" means "do print categories that start with FOO". + + Examples: --filter=-whitespace,+whitespace/braces + --filter=whitespace,runtime/printf,+runtime/printf_format + --filter=-,+build/include_what_you_use + + To see a list of all the categories used in cpplint, pass no arg: + --filter= + + counting=total|toplevel|detailed + The total number of errors found is always printed. If + 'toplevel' is provided, then the count of errors in each of + the top-level categories like 'build' and 'whitespace' will + also be printed. If 'detailed' is provided, then a count + is provided for each category like 'build/class'. + + repository=path + The top level directory of the repository, used to derive the header + guard CPP variable. By default, this is determined by searching for a + path that contains .git, .hg, or .svn. When this flag is specified, the + given path is used instead. This option allows the header guard CPP + variable to remain consistent even if members of a team have different + repository root directories (such as when checking out a subdirectory + with SVN). In addition, users of non-mainstream version control systems + can use this flag to ensure readable header guard CPP variables. + + Examples: + Assuming that Alice checks out ProjectName and Bob checks out + ProjectName/trunk and trunk contains src/chrome/ui/browser.h, then + with no --repository flag, the header guard CPP variable will be: + + Alice => TRUNK_SRC_CHROME_BROWSER_UI_BROWSER_H_ + Bob => SRC_CHROME_BROWSER_UI_BROWSER_H_ + + If Alice uses the --repository=trunk flag and Bob omits the flag or + uses --repository=. then the header guard CPP variable will be: + + Alice => SRC_CHROME_BROWSER_UI_BROWSER_H_ + Bob => SRC_CHROME_BROWSER_UI_BROWSER_H_ + + root=subdir + The root directory used for deriving header guard CPP variable. + This directory is relative to the top level directory of the repository + which by default is determined by searching for a directory that contains + .git, .hg, or .svn but can also be controlled with the --repository flag. + If the specified directory does not exist, this flag is ignored. + + Examples: + Assuming that src is the top level directory of the repository (and + cwd=top/src), the header guard CPP variables for + src/chrome/browser/ui/browser.h are: + + No flag => CHROME_BROWSER_UI_BROWSER_H_ + --root=chrome => BROWSER_UI_BROWSER_H_ + --root=chrome/browser => UI_BROWSER_H_ + --root=.. => SRC_CHROME_BROWSER_UI_BROWSER_H_ + + linelength=digits + This is the allowed line length for the project. The default value is + 80 characters. + + Examples: + --linelength=120 + + recursive + Search for files to lint recursively. Each directory given in the list + of files to be linted is replaced by all files that descend from that + directory. Files with extensions not in the valid extensions list are + excluded. + + exclude=path + Exclude the given path from the list of files to be linted. Relative + paths are evaluated relative to the current directory and shell globbing + is performed. This flag can be provided multiple times to exclude + multiple files. + + Examples: + --exclude=one.cc + --exclude=src/*.cc + --exclude=src/*.cc --exclude=test/*.cc + + extensions=extension,extension,... + The allowed file extensions that cpplint will check + + Examples: + --extensions=%s + + headers=x,y,... + The header extensions that cpplint will treat as .h in checks. Values are + automatically added to --extensions list. + (by default, only files with extensions %s will be assumed to be headers) + + Examples: + --headers=%s + --headers=hpp,hxx + --headers=hpp + + cpplint.py supports per-directory configurations specified in CPPLINT.cfg + files. CPPLINT.cfg file can contain a number of key=value pairs. + Currently the following options are supported: + + set noparent + filter=+filter1,-filter2,... + exclude_files=regex + linelength=80 + root=subdir + headers=x,y,... + + "set noparent" option prevents cpplint from traversing directory tree + upwards looking for more .cfg files in parent directories. This option + is usually placed in the top-level project directory. + + The "filter" option is similar in function to --filter flag. It specifies + message filters in addition to the |_DEFAULT_FILTERS| and those specified + through --filter command-line flag. + + "exclude_files" allows to specify a regular expression to be matched against + a file name. If the expression matches, the file is skipped and not run + through the linter. + + "linelength" allows to specify the allowed line length for the project. + + The "root" option is similar in function to the --root flag (see example + above). Paths are relative to the directory of the CPPLINT.cfg. + + The "headers" option is similar in function to the --headers flag + (see example above). + + CPPLINT.cfg has an effect on files in the same directory and all + sub-directories, unless overridden by a nested configuration file. + + Example file: + filter=-build/include_order,+build/include_alpha + exclude_files=.*\\.cc + + The above example disables build/include_order warning and enables + build/include_alpha as well as excludes all .cc from being + processed by linter, in the current directory (where the .cfg + file is located) and all sub-directories. +""" + +# We categorize each error message we print. Here are the categories. +# We want an explicit list so we can list them all in cpplint --filter=. +# If you add a new error message with a new category, add it to the list +# here! cpplint_unittest.py should tell you if you forget to do this. +_ERROR_CATEGORIES = [ + 'build/class', + 'build/c++11', + 'build/c++14', + 'build/c++tr1', + 'build/deprecated', + 'build/endif_comment', + 'build/explicit_make_pair', + 'build/forward_decl', + 'build/header_guard', + 'build/include', + 'build/include_subdir', + 'build/include_alpha', + 'build/include_order', + 'build/include_what_you_use', + 'build/namespaces_literals', + 'build/namespaces', + 'build/printf_format', + 'build/storage_class', + 'legal/copyright', + 'readability/alt_tokens', + 'readability/braces', + 'readability/casting', + 'readability/check', + 'readability/constructors', + 'readability/fn_size', + 'readability/inheritance', + 'readability/multiline_comment', + 'readability/multiline_string', + 'readability/namespace', + 'readability/nolint', + 'readability/nul', + 'readability/strings', + 'readability/todo', + 'readability/utf8', + 'runtime/arrays', + 'runtime/casting', + 'runtime/explicit', + 'runtime/int', + 'runtime/init', + 'runtime/invalid_increment', + 'runtime/member_string_references', + 'runtime/memset', + 'runtime/indentation_namespace', + 'runtime/operator', + 'runtime/printf', + 'runtime/printf_format', + 'runtime/references', + 'runtime/string', + 'runtime/threadsafe_fn', + 'runtime/vlog', + 'whitespace/blank_line', + 'whitespace/braces', + 'whitespace/comma', + 'whitespace/comments', + 'whitespace/empty_conditional_body', + 'whitespace/empty_if_body', + 'whitespace/empty_loop_body', + 'whitespace/end_of_line', + 'whitespace/ending_newline', + 'whitespace/forcolon', + 'whitespace/indent', + 'whitespace/line_length', + 'whitespace/newline', + 'whitespace/operators', + 'whitespace/parens', + 'whitespace/semicolon', + 'whitespace/tab', + 'whitespace/todo', + ] + +# These error categories are no longer enforced by cpplint, but for backwards- +# compatibility they may still appear in NOLINT comments. +_LEGACY_ERROR_CATEGORIES = [ + 'readability/streams', + 'readability/function', + ] + +# The default state of the category filter. This is overridden by the --filter= +# flag. By default all errors are on, so only add here categories that should be +# off by default (i.e., categories that must be enabled by the --filter= flags). +# All entries here should start with a '-' or '+', as in the --filter= flag. +_DEFAULT_FILTERS = ['-build/include_alpha'] + +# The default list of categories suppressed for C (not C++) files. +_DEFAULT_C_SUPPRESSED_CATEGORIES = [ + 'readability/casting', + ] + +# The default list of categories suppressed for Linux Kernel files. +_DEFAULT_KERNEL_SUPPRESSED_CATEGORIES = [ + 'whitespace/tab', + ] + +# We used to check for high-bit characters, but after much discussion we +# decided those were OK, as long as they were in UTF-8 and didn't represent +# hard-coded international strings, which belong in a separate i18n file. + +# C++ headers +_CPP_HEADERS = frozenset([ + # Legacy + 'algobase.h', + 'algo.h', + 'alloc.h', + 'builtinbuf.h', + 'bvector.h', + 'complex.h', + 'defalloc.h', + 'deque.h', + 'editbuf.h', + 'fstream.h', + 'function.h', + 'hash_map', + 'hash_map.h', + 'hash_set', + 'hash_set.h', + 'hashtable.h', + 'heap.h', + 'indstream.h', + 'iomanip.h', + 'iostream.h', + 'istream.h', + 'iterator.h', + 'list.h', + 'map.h', + 'multimap.h', + 'multiset.h', + 'ostream.h', + 'pair.h', + 'parsestream.h', + 'pfstream.h', + 'procbuf.h', + 'pthread_alloc', + 'pthread_alloc.h', + 'rope', + 'rope.h', + 'ropeimpl.h', + 'set.h', + 'slist', + 'slist.h', + 'stack.h', + 'stdiostream.h', + 'stl_alloc.h', + 'stl_relops.h', + 'streambuf.h', + 'stream.h', + 'strfile.h', + 'strstream.h', + 'tempbuf.h', + 'tree.h', + 'type_traits.h', + 'vector.h', + # 17.6.1.2 C++ library headers + 'algorithm', + 'array', + 'atomic', + 'bitset', + 'chrono', + 'codecvt', + 'complex', + 'condition_variable', + 'deque', + 'exception', + 'forward_list', + 'fstream', + 'functional', + 'future', + 'initializer_list', + 'iomanip', + 'ios', + 'iosfwd', + 'iostream', + 'istream', + 'iterator', + 'limits', + 'list', + 'locale', + 'map', + 'memory', + 'mutex', + 'new', + 'numeric', + 'ostream', + 'queue', + 'random', + 'ratio', + 'regex', + 'scoped_allocator', + 'set', + 'sstream', + 'stack', + 'stdexcept', + 'streambuf', + 'string', + 'strstream', + 'system_error', + 'thread', + 'tuple', + 'typeindex', + 'typeinfo', + 'type_traits', + 'unordered_map', + 'unordered_set', + 'utility', + 'valarray', + 'vector', + # 17.6.1.2 C++14 headers + 'shared_mutex', + # 17.6.1.2 C++17 headers + 'any', + 'charconv', + 'codecvt', + 'execution', + 'filesystem', + 'memory_resource', + 'optional', + 'string_view', + 'variant', + # 17.6.1.2 C++ headers for C library facilities + 'cassert', + 'ccomplex', + 'cctype', + 'cerrno', + 'cfenv', + 'cfloat', + 'cinttypes', + 'ciso646', + 'climits', + 'clocale', + 'cmath', + 'csetjmp', + 'csignal', + 'cstdalign', + 'cstdarg', + 'cstdbool', + 'cstddef', + 'cstdint', + 'cstdio', + 'cstdlib', + 'cstring', + 'ctgmath', + 'ctime', + 'cuchar', + 'cwchar', + 'cwctype', + ]) + +# Type names +_TYPES = re.compile( + r'^(?:' + # [dcl.type.simple] + r'(char(16_t|32_t)?)|wchar_t|' + r'bool|short|int|long|signed|unsigned|float|double|' + # [support.types] + r'(ptrdiff_t|size_t|max_align_t|nullptr_t)|' + # [cstdint.syn] + r'(u?int(_fast|_least)?(8|16|32|64)_t)|' + r'(u?int(max|ptr)_t)|' + r')$') + + +# These headers are excluded from [build/include] and [build/include_order] +# checks: +# - Anything not following google file name conventions (containing an +# uppercase character, such as Python.h or nsStringAPI.h, for example). +# - Lua headers. +_THIRD_PARTY_HEADERS_PATTERN = re.compile( + r'^(?:[^/]*[A-Z][^/]*\.h|lua\.h|lauxlib\.h|lualib\.h)$') + +# Pattern for matching FileInfo.BaseName() against test file name +_test_suffixes = ['_test', '_regtest', '_unittest'] +_TEST_FILE_SUFFIX = '(' + '|'.join(_test_suffixes) + r')$' + +# Pattern that matches only complete whitespace, possibly across multiple lines. +_EMPTY_CONDITIONAL_BODY_PATTERN = re.compile(r'^\s*$', re.DOTALL) + +# Assertion macros. These are defined in base/logging.h and +# testing/base/public/gunit.h. +_CHECK_MACROS = [ + 'DCHECK', 'CHECK', + 'EXPECT_TRUE', 'ASSERT_TRUE', + 'EXPECT_FALSE', 'ASSERT_FALSE', + ] + +# Replacement macros for CHECK/DCHECK/EXPECT_TRUE/EXPECT_FALSE +_CHECK_REPLACEMENT = dict([(macro_var, {}) for macro_var in _CHECK_MACROS]) + +for op, replacement in [('==', 'EQ'), ('!=', 'NE'), + ('>=', 'GE'), ('>', 'GT'), + ('<=', 'LE'), ('<', 'LT')]: + _CHECK_REPLACEMENT['DCHECK'][op] = 'DCHECK_%s' % replacement + _CHECK_REPLACEMENT['CHECK'][op] = 'CHECK_%s' % replacement + _CHECK_REPLACEMENT['EXPECT_TRUE'][op] = 'EXPECT_%s' % replacement + _CHECK_REPLACEMENT['ASSERT_TRUE'][op] = 'ASSERT_%s' % replacement + +for op, inv_replacement in [('==', 'NE'), ('!=', 'EQ'), + ('>=', 'LT'), ('>', 'LE'), + ('<=', 'GT'), ('<', 'GE')]: + _CHECK_REPLACEMENT['EXPECT_FALSE'][op] = 'EXPECT_%s' % inv_replacement + _CHECK_REPLACEMENT['ASSERT_FALSE'][op] = 'ASSERT_%s' % inv_replacement + +# Alternative tokens and their replacements. For full list, see section 2.5 +# Alternative tokens [lex.digraph] in the C++ standard. +# +# Digraphs (such as '%:') are not included here since it's a mess to +# match those on a word boundary. +_ALT_TOKEN_REPLACEMENT = { + 'and': '&&', + 'bitor': '|', + 'or': '||', + 'xor': '^', + 'compl': '~', + 'bitand': '&', + 'and_eq': '&=', + 'or_eq': '|=', + 'xor_eq': '^=', + 'not': '!', + 'not_eq': '!=' + } + +# Compile regular expression that matches all the above keywords. The "[ =()]" +# bit is meant to avoid matching these keywords outside of boolean expressions. +# +# False positives include C-style multi-line comments and multi-line strings +# but those have always been troublesome for cpplint. +_ALT_TOKEN_REPLACEMENT_PATTERN = re.compile( + r'[ =()](' + ('|'.join(_ALT_TOKEN_REPLACEMENT.keys())) + r')(?=[ (]|$)') + + +# These constants define types of headers for use with +# _IncludeState.CheckNextIncludeOrder(). +_C_SYS_HEADER = 1 +_CPP_SYS_HEADER = 2 +_LIKELY_MY_HEADER = 3 +_POSSIBLE_MY_HEADER = 4 +_OTHER_HEADER = 5 + +# These constants define the current inline assembly state +_NO_ASM = 0 # Outside of inline assembly block +_INSIDE_ASM = 1 # Inside inline assembly block +_END_ASM = 2 # Last line of inline assembly block +_BLOCK_ASM = 3 # The whole block is an inline assembly block + +# Match start of assembly blocks +_MATCH_ASM = re.compile(r'^\s*(?:asm|_asm|__asm|__asm__)' + r'(?:\s+(volatile|__volatile__))?' + r'\s*[{(]') + +# Match strings that indicate we're working on a C (not C++) file. +_SEARCH_C_FILE = re.compile(r'\b(?:LINT_C_FILE|' + r'vim?:\s*.*(\s*|:)filetype=c(\s*|:|$))') + +# Match string that indicates we're working on a Linux Kernel file. +_SEARCH_KERNEL_FILE = re.compile(r'\b(?:LINT_KERNEL_FILE)') + +_regexp_compile_cache = {} + +# {str, set(int)}: a map from error categories to sets of linenumbers +# on which those errors are expected and should be suppressed. +_error_suppressions = {} + +# The root directory used for deriving header guard CPP variable. +# This is set by --root flag. +_root = None +_root_debug = False + +# The top level repository directory. If set, _root is calculated relative to +# this directory instead of the directory containing version control artifacts. +# This is set by the --repository flag. +_repository = None + +# Files to exclude from linting. This is set by the --exclude flag. +_excludes = None + +# Whether to supress PrintInfo messages +_quiet = False + +# The allowed line length of files. +# This is set by --linelength flag. +_line_length = 80 + +try: + unicode +except NameError: + # -- pylint: disable=redefined-builtin + basestring = unicode = str + +try: + long +except NameError: + # -- pylint: disable=redefined-builtin + long = int + +if sys.version_info < (3,): + # -- pylint: disable=no-member + # BINARY_TYPE = str + itervalues = dict.itervalues + iteritems = dict.iteritems +else: + # BINARY_TYPE = bytes + itervalues = dict.values + iteritems = dict.items + +def unicode_escape_decode(x): + if sys.version_info < (3,): + return codecs.unicode_escape_decode(x)[0] + else: + return x + +# Treat all headers starting with 'h' equally: .h, .hpp, .hxx etc. +# This is set by --headers flag. +_hpp_headers = set(['h', 'hh', 'hpp', 'hxx', 'h++', 'cuh']) + +# {str, bool}: a map from error categories to booleans which indicate if the +# category should be suppressed for every line. +_global_error_suppressions = {} + +def ProcessHppHeadersOption(val): + global _hpp_headers + try: + _hpp_headers = set(val.split(',')) + # Automatically append to extensions list so it does not have to be set 2 times + _valid_extensions.update(_hpp_headers) + except ValueError: + PrintUsage('Header extensions must be comma separated list.') + +def IsHeaderExtension(file_extension): + return file_extension in _hpp_headers + +def GetHeaderExtensions(): + return _hpp_headers or ['h'] + +# The allowed extensions for file names +# This is set by --extensions flag +def GetAllExtensions(): + if not _valid_extensions: + return GetHeaderExtensions().union(set(['c', 'cc', 'cpp', 'cxx', 'c++', 'cu'])) + return _valid_extensions + +def GetNonHeaderExtensions(): + return GetAllExtensions().difference(GetHeaderExtensions()) + + + +def ParseNolintSuppressions(filename, raw_line, linenum, error): + """Updates the global list of line error-suppressions. + + Parses any NOLINT comments on the current line, updating the global + error_suppressions store. Reports an error if the NOLINT comment + was malformed. + + Args: + filename: str, the name of the input file. + raw_line: str, the line of input text, with comments. + linenum: int, the number of the current line. + error: function, an error handler. + """ + matched = Search(r'\bNOLINT(NEXTLINE)?\b(\([^)]+\))?', raw_line) + if matched: + if matched.group(1): + suppressed_line = linenum + 1 + else: + suppressed_line = linenum + category = matched.group(2) + if category in (None, '(*)'): # => "suppress all" + _error_suppressions.setdefault(None, set()).add(suppressed_line) + else: + if category.startswith('(') and category.endswith(')'): + category = category[1:-1] + if category in _ERROR_CATEGORIES: + _error_suppressions.setdefault(category, set()).add(suppressed_line) + elif category not in _LEGACY_ERROR_CATEGORIES: + error(filename, linenum, 'readability/nolint', 5, + 'Unknown NOLINT error category: %s' % category) + + +def ProcessGlobalSuppresions(lines): + """Updates the list of global error suppressions. + + Parses any lint directives in the file that have global effect. + + Args: + lines: An array of strings, each representing a line of the file, with the + last element being empty if the file is terminated with a newline. + """ + for line in lines: + if _SEARCH_C_FILE.search(line): + for category in _DEFAULT_C_SUPPRESSED_CATEGORIES: + _global_error_suppressions[category] = True + if _SEARCH_KERNEL_FILE.search(line): + for category in _DEFAULT_KERNEL_SUPPRESSED_CATEGORIES: + _global_error_suppressions[category] = True + + +def ResetNolintSuppressions(): + """Resets the set of NOLINT suppressions to empty.""" + _error_suppressions.clear() + _global_error_suppressions.clear() + + +def IsErrorSuppressedByNolint(category, linenum): + """Returns true if the specified error category is suppressed on this line. + + Consults the global error_suppressions map populated by + ParseNolintSuppressions/ProcessGlobalSuppresions/ResetNolintSuppressions. + + Args: + category: str, the category of the error. + linenum: int, the current line number. + Returns: + bool, True iff the error should be suppressed due to a NOLINT comment or + global suppression. + """ + return (_global_error_suppressions.get(category, False) or + linenum in _error_suppressions.get(category, set()) or + linenum in _error_suppressions.get(None, set())) + + +def Match(pattern, s): + """Matches the string with the pattern, caching the compiled regexp.""" + # The regexp compilation caching is inlined in both Match and Search for + # performance reasons; factoring it out into a separate function turns out + # to be noticeably expensive. + if pattern not in _regexp_compile_cache: + _regexp_compile_cache[pattern] = sre_compile.compile(pattern) + return _regexp_compile_cache[pattern].match(s) + + +def ReplaceAll(pattern, rep, s): + """Replaces instances of pattern in a string with a replacement. + + The compiled regex is kept in a cache shared by Match and Search. + + Args: + pattern: regex pattern + rep: replacement text + s: search string + + Returns: + string with replacements made (or original string if no replacements) + """ + if pattern not in _regexp_compile_cache: + _regexp_compile_cache[pattern] = sre_compile.compile(pattern) + return _regexp_compile_cache[pattern].sub(rep, s) + + +def Search(pattern, s): + """Searches the string for the pattern, caching the compiled regexp.""" + if pattern not in _regexp_compile_cache: + _regexp_compile_cache[pattern] = sre_compile.compile(pattern) + return _regexp_compile_cache[pattern].search(s) + + +def _IsSourceExtension(s): + """File extension (excluding dot) matches a source file extension.""" + return s in GetNonHeaderExtensions() + + +class _IncludeState(object): + """Tracks line numbers for includes, and the order in which includes appear. + + include_list contains list of lists of (header, line number) pairs. + It's a lists of lists rather than just one flat list to make it + easier to update across preprocessor boundaries. + + Call CheckNextIncludeOrder() once for each header in the file, passing + in the type constants defined above. Calls in an illegal order will + raise an _IncludeError with an appropriate error message. + + """ + # self._section will move monotonically through this set. If it ever + # needs to move backwards, CheckNextIncludeOrder will raise an error. + _INITIAL_SECTION = 0 + _MY_H_SECTION = 1 + _C_SECTION = 2 + _CPP_SECTION = 3 + _OTHER_H_SECTION = 4 + + _TYPE_NAMES = { + _C_SYS_HEADER: 'C system header', + _CPP_SYS_HEADER: 'C++ system header', + _LIKELY_MY_HEADER: 'header this file implements', + _POSSIBLE_MY_HEADER: 'header this file may implement', + _OTHER_HEADER: 'other header', + } + _SECTION_NAMES = { + _INITIAL_SECTION: "... nothing. (This can't be an error.)", + _MY_H_SECTION: 'a header this file implements', + _C_SECTION: 'C system header', + _CPP_SECTION: 'C++ system header', + _OTHER_H_SECTION: 'other header', + } + + def __init__(self): + self.include_list = [[]] + self._section = None + self._last_header = None + self.ResetSection('') + + def FindHeader(self, header): + """Check if a header has already been included. + + Args: + header: header to check. + Returns: + Line number of previous occurrence, or -1 if the header has not + been seen before. + """ + for section_list in self.include_list: + for f in section_list: + if f[0] == header: + return f[1] + return -1 + + def ResetSection(self, directive): + """Reset section checking for preprocessor directive. + + Args: + directive: preprocessor directive (e.g. "if", "else"). + """ + # The name of the current section. + self._section = self._INITIAL_SECTION + # The path of last found header. + self._last_header = '' + + # Update list of includes. Note that we never pop from the + # include list. + if directive in ('if', 'ifdef', 'ifndef'): + self.include_list.append([]) + elif directive in ('else', 'elif'): + self.include_list[-1] = [] + + def SetLastHeader(self, header_path): + self._last_header = header_path + + def CanonicalizeAlphabeticalOrder(self, header_path): + """Returns a path canonicalized for alphabetical comparison. + + - replaces "-" with "_" so they both cmp the same. + - removes '-inl' since we don't require them to be after the main header. + - lowercase everything, just in case. + + Args: + header_path: Path to be canonicalized. + + Returns: + Canonicalized path. + """ + return header_path.replace('-inl.h', '.h').replace('-', '_').lower() + + def IsInAlphabeticalOrder(self, clean_lines, linenum, header_path): + """Check if a header is in alphabetical order with the previous header. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + header_path: Canonicalized header to be checked. + + Returns: + Returns true if the header is in alphabetical order. + """ + # If previous section is different from current section, _last_header will + # be reset to empty string, so it's always less than current header. + # + # If previous line was a blank line, assume that the headers are + # intentionally sorted the way they are. + if (self._last_header > header_path and + Match(r'^\s*#\s*include\b', clean_lines.elided[linenum - 1])): + return False + return True + + def CheckNextIncludeOrder(self, header_type): + """Returns a non-empty error message if the next header is out of order. + + This function also updates the internal state to be ready to check + the next include. + + Args: + header_type: One of the _XXX_HEADER constants defined above. + + Returns: + The empty string if the header is in the right order, or an + error message describing what's wrong. + + """ + error_message = ('Found %s after %s' % + (self._TYPE_NAMES[header_type], + self._SECTION_NAMES[self._section])) + + last_section = self._section + + if header_type == _C_SYS_HEADER: + if self._section <= self._C_SECTION: + self._section = self._C_SECTION + else: + self._last_header = '' + return error_message + elif header_type == _CPP_SYS_HEADER: + if self._section <= self._CPP_SECTION: + self._section = self._CPP_SECTION + else: + self._last_header = '' + return error_message + elif header_type == _LIKELY_MY_HEADER: + if self._section <= self._MY_H_SECTION: + self._section = self._MY_H_SECTION + else: + self._section = self._OTHER_H_SECTION + elif header_type == _POSSIBLE_MY_HEADER: + if self._section <= self._MY_H_SECTION: + self._section = self._MY_H_SECTION + else: + # This will always be the fallback because we're not sure + # enough that the header is associated with this file. + self._section = self._OTHER_H_SECTION + else: + assert header_type == _OTHER_HEADER + self._section = self._OTHER_H_SECTION + + if last_section != self._section: + self._last_header = '' + + return '' + + +class _CppLintState(object): + """Maintains module-wide state..""" + + def __init__(self): + self.verbose_level = 1 # global setting. + self.error_count = 0 # global count of reported errors + # filters to apply when emitting error messages + self.filters = _DEFAULT_FILTERS[:] + # backup of filter list. Used to restore the state after each file. + self._filters_backup = self.filters[:] + self.counting = 'total' # In what way are we counting errors? + self.errors_by_category = {} # string to int dict storing error counts + self.quiet = False # Suppress non-error messagess? + + # output format: + # "emacs" - format that emacs can parse (default) + # "eclipse" - format that eclipse can parse + # "vs7" - format that Microsoft Visual Studio 7 can parse + # "junit" - format that Jenkins, Bamboo, etc can parse + self.output_format = 'emacs' + + # For JUnit output, save errors and failures until the end so that they + # can be written into the XML + self._junit_errors = [] + self._junit_failures = [] + + def SetOutputFormat(self, output_format): + """Sets the output format for errors.""" + self.output_format = output_format + + def SetQuiet(self, quiet): + """Sets the module's quiet settings, and returns the previous setting.""" + last_quiet = self.quiet + self.quiet = quiet + return last_quiet + + def SetVerboseLevel(self, level): + """Sets the module's verbosity, and returns the previous setting.""" + last_verbose_level = self.verbose_level + self.verbose_level = level + return last_verbose_level + + def SetCountingStyle(self, counting_style): + """Sets the module's counting options.""" + self.counting = counting_style + + def SetFilters(self, filters): + """Sets the error-message filters. + + These filters are applied when deciding whether to emit a given + error message. + + Args: + filters: A string of comma-separated filters (eg "+whitespace/indent"). + Each filter should start with + or -; else we die. + + Raises: + ValueError: The comma-separated filters did not all start with '+' or '-'. + E.g. "-,+whitespace,-whitespace/indent,whitespace/badfilter" + """ + # Default filters always have less priority than the flag ones. + self.filters = _DEFAULT_FILTERS[:] + self.AddFilters(filters) + + def AddFilters(self, filters): + """ Adds more filters to the existing list of error-message filters. """ + for filt in filters.split(','): + clean_filt = filt.strip() + if clean_filt: + self.filters.append(clean_filt) + for filt in self.filters: + if not (filt.startswith('+') or filt.startswith('-')): + raise ValueError('Every filter in --filters must start with + or -' + ' (%s does not)' % filt) + + def BackupFilters(self): + """ Saves the current filter list to backup storage.""" + self._filters_backup = self.filters[:] + + def RestoreFilters(self): + """ Restores filters previously backed up.""" + self.filters = self._filters_backup[:] + + def ResetErrorCounts(self): + """Sets the module's error statistic back to zero.""" + self.error_count = 0 + self.errors_by_category = {} + + def IncrementErrorCount(self, category): + """Bumps the module's error statistic.""" + self.error_count += 1 + if self.counting in ('toplevel', 'detailed'): + if self.counting != 'detailed': + category = category.split('/')[0] + if category not in self.errors_by_category: + self.errors_by_category[category] = 0 + self.errors_by_category[category] += 1 + + def PrintErrorCounts(self): + """Print a summary of errors by category, and the total.""" + for category, count in sorted(iteritems(self.errors_by_category)): + self.PrintInfo('Category \'%s\' errors found: %d\n' % + (category, count)) + if self.error_count > 0: + self.PrintInfo('Total errors found: %d\n' % self.error_count) + + def PrintInfo(self, message): + if not _quiet and self.output_format != 'junit': + sys.stdout.write(message) + + def PrintError(self, message): + if self.output_format == 'junit': + self._junit_errors.append(message) + else: + sys.stderr.write(message) + + def AddJUnitFailure(self, filename, linenum, message, category, confidence): + self._junit_failures.append((filename, linenum, message, category, + confidence)) + + def FormatJUnitXML(self): + num_errors = len(self._junit_errors) + num_failures = len(self._junit_failures) + + testsuite = xml.etree.ElementTree.Element('testsuite') + testsuite.attrib['name'] = 'cpplint' + testsuite.attrib['errors'] = str(num_errors) + testsuite.attrib['failures'] = str(num_failures) + + if num_errors == 0 and num_failures == 0: + testsuite.attrib['tests'] = str(1) + xml.etree.ElementTree.SubElement(testsuite, 'testcase', name='passed') + + else: + testsuite.attrib['tests'] = str(num_errors + num_failures) + if num_errors > 0: + testcase = xml.etree.ElementTree.SubElement(testsuite, 'testcase') + testcase.attrib['name'] = 'errors' + error = xml.etree.ElementTree.SubElement(testcase, 'error') + error.text = '\n'.join(self._junit_errors) + if num_failures > 0: + # Group failures by file + failed_file_order = [] + failures_by_file = {} + for failure in self._junit_failures: + failed_file = failure[0] + if failed_file not in failed_file_order: + failed_file_order.append(failed_file) + failures_by_file[failed_file] = [] + failures_by_file[failed_file].append(failure) + # Create a testcase for each file + for failed_file in failed_file_order: + failures = failures_by_file[failed_file] + testcase = xml.etree.ElementTree.SubElement(testsuite, 'testcase') + testcase.attrib['name'] = failed_file + failure = xml.etree.ElementTree.SubElement(testcase, 'failure') + template = '{0}: {1} [{2}] [{3}]' + texts = [template.format(f[1], f[2], f[3], f[4]) for f in failures] + failure.text = '\n'.join(texts) + + xml_decl = '\n' + return xml_decl + xml.etree.ElementTree.tostring(testsuite, 'utf-8').decode('utf-8') + + +_cpplint_state = _CppLintState() + + +def _OutputFormat(): + """Gets the module's output format.""" + return _cpplint_state.output_format + + +def _SetOutputFormat(output_format): + """Sets the module's output format.""" + _cpplint_state.SetOutputFormat(output_format) + +def _Quiet(): + """Return's the module's quiet setting.""" + return _cpplint_state.quiet + +def _SetQuiet(quiet): + """Set the module's quiet status, and return previous setting.""" + return _cpplint_state.SetQuiet(quiet) + + +def _VerboseLevel(): + """Returns the module's verbosity setting.""" + return _cpplint_state.verbose_level + + +def _SetVerboseLevel(level): + """Sets the module's verbosity, and returns the previous setting.""" + return _cpplint_state.SetVerboseLevel(level) + + +def _SetCountingStyle(level): + """Sets the module's counting options.""" + _cpplint_state.SetCountingStyle(level) + + +def _Filters(): + """Returns the module's list of output filters, as a list.""" + return _cpplint_state.filters + + +def _SetFilters(filters): + """Sets the module's error-message filters. + + These filters are applied when deciding whether to emit a given + error message. + + Args: + filters: A string of comma-separated filters (eg "whitespace/indent"). + Each filter should start with + or -; else we die. + """ + _cpplint_state.SetFilters(filters) + +def _AddFilters(filters): + """Adds more filter overrides. + + Unlike _SetFilters, this function does not reset the current list of filters + available. + + Args: + filters: A string of comma-separated filters (eg "whitespace/indent"). + Each filter should start with + or -; else we die. + """ + _cpplint_state.AddFilters(filters) + +def _BackupFilters(): + """ Saves the current filter list to backup storage.""" + _cpplint_state.BackupFilters() + +def _RestoreFilters(): + """ Restores filters previously backed up.""" + _cpplint_state.RestoreFilters() + +class _FunctionState(object): + """Tracks current function name and the number of lines in its body.""" + + _NORMAL_TRIGGER = 250 # for --v=0, 500 for --v=1, etc. + _TEST_TRIGGER = 400 # about 50% more than _NORMAL_TRIGGER. + + def __init__(self): + self.in_a_function = False + self.lines_in_function = 0 + self.current_function = '' + + def Begin(self, function_name): + """Start analyzing function body. + + Args: + function_name: The name of the function being tracked. + """ + self.in_a_function = True + self.lines_in_function = 0 + self.current_function = function_name + + def Count(self): + """Count line in current function body.""" + if self.in_a_function: + self.lines_in_function += 1 + + def Check(self, error, filename, linenum): + """Report if too many lines in function body. + + Args: + error: The function to call with any errors found. + filename: The name of the current file. + linenum: The number of the line to check. + """ + if not self.in_a_function: + return + + if Match(r'T(EST|est)', self.current_function): + base_trigger = self._TEST_TRIGGER + else: + base_trigger = self._NORMAL_TRIGGER + trigger = base_trigger * 2**_VerboseLevel() + + if self.lines_in_function > trigger: + error_level = int(math.log(self.lines_in_function / base_trigger, 2)) + # 50 => 0, 100 => 1, 200 => 2, 400 => 3, 800 => 4, 1600 => 5, ... + if error_level > 5: + error_level = 5 + error(filename, linenum, 'readability/fn_size', error_level, + 'Small and focused functions are preferred:' + ' %s has %d non-comment lines' + ' (error triggered by exceeding %d lines).' % ( + self.current_function, self.lines_in_function, trigger)) + + def End(self): + """Stop analyzing function body.""" + self.in_a_function = False + + +class _IncludeError(Exception): + """Indicates a problem with the include order in a file.""" + pass + + +class FileInfo(object): + """Provides utility functions for filenames. + + FileInfo provides easy access to the components of a file's path + relative to the project root. + """ + + def __init__(self, filename): + self._filename = filename + + def FullName(self): + """Make Windows paths like Unix.""" + return os.path.abspath(self._filename).replace('\\', '/') + + def RepositoryName(self): + r"""FullName after removing the local path to the repository. + + If we have a real absolute path name here we can try to do something smart: + detecting the root of the checkout and truncating /path/to/checkout from + the name so that we get header guards that don't include things like + "C:\Documents and Settings\..." or "/home/username/..." in them and thus + people on different computers who have checked the source out to different + locations won't see bogus errors. + """ + fullname = self.FullName() + + if os.path.exists(fullname): + project_dir = os.path.dirname(fullname) + + # If the user specified a repository path, it exists, and the file is + # contained in it, use the specified repository path + if _repository: + repo = FileInfo(_repository).FullName() + root_dir = project_dir + while os.path.exists(root_dir): + # allow case insensitive compare on Windows + if os.path.normcase(root_dir) == os.path.normcase(repo): + return os.path.relpath(fullname, root_dir).replace('\\', '/') + one_up_dir = os.path.dirname(root_dir) + if one_up_dir == root_dir: + break + root_dir = one_up_dir + + if os.path.exists(os.path.join(project_dir, ".svn")): + # If there's a .svn file in the current directory, we recursively look + # up the directory tree for the top of the SVN checkout + root_dir = project_dir + one_up_dir = os.path.dirname(root_dir) + while os.path.exists(os.path.join(one_up_dir, ".svn")): + root_dir = os.path.dirname(root_dir) + one_up_dir = os.path.dirname(one_up_dir) + + prefix = os.path.commonprefix([root_dir, project_dir]) + return fullname[len(prefix) + 1:] + + # Not SVN <= 1.6? Try to find a git, hg, or svn top level directory by + # searching up from the current path. + root_dir = current_dir = os.path.dirname(fullname) + while current_dir != os.path.dirname(current_dir): + if (os.path.exists(os.path.join(current_dir, ".git")) or + os.path.exists(os.path.join(current_dir, ".hg")) or + os.path.exists(os.path.join(current_dir, ".svn"))): + root_dir = current_dir + current_dir = os.path.dirname(current_dir) + + if (os.path.exists(os.path.join(root_dir, ".git")) or + os.path.exists(os.path.join(root_dir, ".hg")) or + os.path.exists(os.path.join(root_dir, ".svn"))): + prefix = os.path.commonprefix([root_dir, project_dir]) + return fullname[len(prefix) + 1:] + + # Don't know what to do; header guard warnings may be wrong... + return fullname + + def Split(self): + """Splits the file into the directory, basename, and extension. + + For 'chrome/browser/browser.cc', Split() would + return ('chrome/browser', 'browser', '.cc') + + Returns: + A tuple of (directory, basename, extension). + """ + + googlename = self.RepositoryName() + project, rest = os.path.split(googlename) + return (project,) + os.path.splitext(rest) + + def BaseName(self): + """File base name - text after the final slash, before the final period.""" + return self.Split()[1] + + def Extension(self): + """File extension - text following the final period, includes that period.""" + return self.Split()[2] + + def NoExtension(self): + """File has no source file extension.""" + return '/'.join(self.Split()[0:2]) + + def IsSource(self): + """File has a source file extension.""" + return _IsSourceExtension(self.Extension()[1:]) + + +def _ShouldPrintError(category, confidence, linenum): + """If confidence >= verbose, category passes filter and is not suppressed.""" + + # There are three ways we might decide not to print an error message: + # a "NOLINT(category)" comment appears in the source, + # the verbosity level isn't high enough, or the filters filter it out. + if IsErrorSuppressedByNolint(category, linenum): + return False + + if confidence < _cpplint_state.verbose_level: + return False + + is_filtered = False + for one_filter in _Filters(): + if one_filter.startswith('-'): + if category.startswith(one_filter[1:]): + is_filtered = True + elif one_filter.startswith('+'): + if category.startswith(one_filter[1:]): + is_filtered = False + else: + assert False # should have been checked for in SetFilter. + if is_filtered: + return False + + return True + + +def Error(filename, linenum, category, confidence, message): + """Logs the fact we've found a lint error. + + We log where the error was found, and also our confidence in the error, + that is, how certain we are this is a legitimate style regression, and + not a misidentification or a use that's sometimes justified. + + False positives can be suppressed by the use of + "cpplint(category)" comments on the offending line. These are + parsed into _error_suppressions. + + Args: + filename: The name of the file containing the error. + linenum: The number of the line containing the error. + category: A string used to describe the "category" this bug + falls under: "whitespace", say, or "runtime". Categories + may have a hierarchy separated by slashes: "whitespace/indent". + confidence: A number from 1-5 representing a confidence score for + the error, with 5 meaning that we are certain of the problem, + and 1 meaning that it could be a legitimate construct. + message: The error message. + """ + if _ShouldPrintError(category, confidence, linenum): + _cpplint_state.IncrementErrorCount(category) + if _cpplint_state.output_format == 'vs7': + _cpplint_state.PrintError('%s(%s): error cpplint: [%s] %s [%d]\n' % ( + filename, linenum, category, message, confidence)) + elif _cpplint_state.output_format == 'eclipse': + sys.stderr.write('%s:%s: warning: %s [%s] [%d]\n' % ( + filename, linenum, message, category, confidence)) + elif _cpplint_state.output_format == 'junit': + _cpplint_state.AddJUnitFailure(filename, linenum, message, category, + confidence) + else: + final_message = '%s:%s: %s [%s] [%d]\n' % ( + filename, linenum, message, category, confidence) + sys.stderr.write(final_message) + +# Matches standard C++ escape sequences per 2.13.2.3 of the C++ standard. +_RE_PATTERN_CLEANSE_LINE_ESCAPES = re.compile( + r'\\([abfnrtv?"\\\']|\d+|x[0-9a-fA-F]+)') +# Match a single C style comment on the same line. +_RE_PATTERN_C_COMMENTS = r'/\*(?:[^*]|\*(?!/))*\*/' +# Matches multi-line C style comments. +# This RE is a little bit more complicated than one might expect, because we +# have to take care of space removals tools so we can handle comments inside +# statements better. +# The current rule is: We only clear spaces from both sides when we're at the +# end of the line. Otherwise, we try to remove spaces from the right side, +# if this doesn't work we try on left side but only if there's a non-character +# on the right. +_RE_PATTERN_CLEANSE_LINE_C_COMMENTS = re.compile( + r'(\s*' + _RE_PATTERN_C_COMMENTS + r'\s*$|' + + _RE_PATTERN_C_COMMENTS + r'\s+|' + + r'\s+' + _RE_PATTERN_C_COMMENTS + r'(?=\W)|' + + _RE_PATTERN_C_COMMENTS + r')') + + +def IsCppString(line): + """Does line terminate so, that the next symbol is in string constant. + + This function does not consider single-line nor multi-line comments. + + Args: + line: is a partial line of code starting from the 0..n. + + Returns: + True, if next character appended to 'line' is inside a + string constant. + """ + + line = line.replace(r'\\', 'XX') # after this, \\" does not match to \" + return ((line.count('"') - line.count(r'\"') - line.count("'\"'")) & 1) == 1 + + +def CleanseRawStrings(raw_lines): + """Removes C++11 raw strings from lines. + + Before: + static const char kData[] = R"( + multi-line string + )"; + + After: + static const char kData[] = "" + (replaced by blank line) + ""; + + Args: + raw_lines: list of raw lines. + + Returns: + list of lines with C++11 raw strings replaced by empty strings. + """ + + delimiter = None + lines_without_raw_strings = [] + for line in raw_lines: + if delimiter: + # Inside a raw string, look for the end + end = line.find(delimiter) + if end >= 0: + # Found the end of the string, match leading space for this + # line and resume copying the original lines, and also insert + # a "" on the last line. + leading_space = Match(r'^(\s*)\S', line) + line = leading_space.group(1) + '""' + line[end + len(delimiter):] + delimiter = None + else: + # Haven't found the end yet, append a blank line. + line = '""' + + # Look for beginning of a raw string, and replace them with + # empty strings. This is done in a loop to handle multiple raw + # strings on the same line. + while delimiter is None: + # Look for beginning of a raw string. + # See 2.14.15 [lex.string] for syntax. + # + # Once we have matched a raw string, we check the prefix of the + # line to make sure that the line is not part of a single line + # comment. It's done this way because we remove raw strings + # before removing comments as opposed to removing comments + # before removing raw strings. This is because there are some + # cpplint checks that requires the comments to be preserved, but + # we don't want to check comments that are inside raw strings. + matched = Match(r'^(.*?)\b(?:R|u8R|uR|UR|LR)"([^\s\\()]*)\((.*)$', line) + if (matched and + not Match(r'^([^\'"]|\'(\\.|[^\'])*\'|"(\\.|[^"])*")*//', + matched.group(1))): + delimiter = ')' + matched.group(2) + '"' + + end = matched.group(3).find(delimiter) + if end >= 0: + # Raw string ended on same line + line = (matched.group(1) + '""' + + matched.group(3)[end + len(delimiter):]) + delimiter = None + else: + # Start of a multi-line raw string + line = matched.group(1) + '""' + else: + break + + lines_without_raw_strings.append(line) + + # TODO(unknown): if delimiter is not None here, we might want to + # emit a warning for unterminated string. + return lines_without_raw_strings + + +def FindNextMultiLineCommentStart(lines, lineix): + """Find the beginning marker for a multiline comment.""" + while lineix < len(lines): + if lines[lineix].strip().startswith('/*'): + # Only return this marker if the comment goes beyond this line + if lines[lineix].strip().find('*/', 2) < 0: + return lineix + lineix += 1 + return len(lines) + + +def FindNextMultiLineCommentEnd(lines, lineix): + """We are inside a comment, find the end marker.""" + while lineix < len(lines): + if lines[lineix].strip().endswith('*/'): + return lineix + lineix += 1 + return len(lines) + + +def RemoveMultiLineCommentsFromRange(lines, begin, end): + """Clears a range of lines for multi-line comments.""" + # Having // dummy comments makes the lines non-empty, so we will not get + # unnecessary blank line warnings later in the code. + for i in range(begin, end): + lines[i] = '/**/' + + +def RemoveMultiLineComments(filename, lines, error): + """Removes multiline (c-style) comments from lines.""" + lineix = 0 + while lineix < len(lines): + lineix_begin = FindNextMultiLineCommentStart(lines, lineix) + if lineix_begin >= len(lines): + return + lineix_end = FindNextMultiLineCommentEnd(lines, lineix_begin) + if lineix_end >= len(lines): + error(filename, lineix_begin + 1, 'readability/multiline_comment', 5, + 'Could not find end of multi-line comment') + return + RemoveMultiLineCommentsFromRange(lines, lineix_begin, lineix_end + 1) + lineix = lineix_end + 1 + + +def CleanseComments(line): + """Removes //-comments and single-line C-style /* */ comments. + + Args: + line: A line of C++ source. + + Returns: + The line with single-line comments removed. + """ + commentpos = line.find('//') + if commentpos != -1 and not IsCppString(line[:commentpos]): + line = line[:commentpos].rstrip() + # get rid of /* ... */ + return _RE_PATTERN_CLEANSE_LINE_C_COMMENTS.sub('', line) + + +class CleansedLines(object): + """Holds 4 copies of all lines with different preprocessing applied to them. + + 1) elided member contains lines without strings and comments. + 2) lines member contains lines without comments. + 3) raw_lines member contains all the lines without processing. + 4) lines_without_raw_strings member is same as raw_lines, but with C++11 raw + strings removed. + All these members are of , and of the same length. + """ + + def __init__(self, lines): + self.elided = [] + self.lines = [] + self.raw_lines = lines + self.num_lines = len(lines) + self.lines_without_raw_strings = CleanseRawStrings(lines) + for linenum in range(len(self.lines_without_raw_strings)): + self.lines.append(CleanseComments( + self.lines_without_raw_strings[linenum])) + elided = self._CollapseStrings(self.lines_without_raw_strings[linenum]) + self.elided.append(CleanseComments(elided)) + + def NumLines(self): + """Returns the number of lines represented.""" + return self.num_lines + + @staticmethod + def _CollapseStrings(elided): + """Collapses strings and chars on a line to simple "" or '' blocks. + + We nix strings first so we're not fooled by text like '"http://"' + + Args: + elided: The line being processed. + + Returns: + The line with collapsed strings. + """ + if _RE_PATTERN_INCLUDE.match(elided): + return elided + + # Remove escaped characters first to make quote/single quote collapsing + # basic. Things that look like escaped characters shouldn't occur + # outside of strings and chars. + elided = _RE_PATTERN_CLEANSE_LINE_ESCAPES.sub('', elided) + + # Replace quoted strings and digit separators. Both single quotes + # and double quotes are processed in the same loop, otherwise + # nested quotes wouldn't work. + collapsed = '' + while True: + # Find the first quote character + match = Match(r'^([^\'"]*)([\'"])(.*)$', elided) + if not match: + collapsed += elided + break + head, quote, tail = match.groups() + + if quote == '"': + # Collapse double quoted strings + second_quote = tail.find('"') + if second_quote >= 0: + collapsed += head + '""' + elided = tail[second_quote + 1:] + else: + # Unmatched double quote, don't bother processing the rest + # of the line since this is probably a multiline string. + collapsed += elided + break + else: + # Found single quote, check nearby text to eliminate digit separators. + # + # There is no special handling for floating point here, because + # the integer/fractional/exponent parts would all be parsed + # correctly as long as there are digits on both sides of the + # separator. So we are fine as long as we don't see something + # like "0.'3" (gcc 4.9.0 will not allow this literal). + if Search(r'\b(?:0[bBxX]?|[1-9])[0-9a-fA-F]*$', head): + match_literal = Match(r'^((?:\'?[0-9a-zA-Z_])*)(.*)$', "'" + tail) + collapsed += head + match_literal.group(1).replace("'", '') + elided = match_literal.group(2) + else: + second_quote = tail.find('\'') + if second_quote >= 0: + collapsed += head + "''" + elided = tail[second_quote + 1:] + else: + # Unmatched single quote + collapsed += elided + break + + return collapsed + + +def FindEndOfExpressionInLine(line, startpos, stack): + """Find the position just after the end of current parenthesized expression. + + Args: + line: a CleansedLines line. + startpos: start searching at this position. + stack: nesting stack at startpos. + + Returns: + On finding matching end: (index just after matching end, None) + On finding an unclosed expression: (-1, None) + Otherwise: (-1, new stack at end of this line) + """ + for i in xrange(startpos, len(line)): + char = line[i] + if char in '([{': + # Found start of parenthesized expression, push to expression stack + stack.append(char) + elif char == '<': + # Found potential start of template argument list + if i > 0 and line[i - 1] == '<': + # Left shift operator + if stack and stack[-1] == '<': + stack.pop() + if not stack: + return (-1, None) + elif i > 0 and Search(r'\boperator\s*$', line[0:i]): + # operator<, don't add to stack + continue + else: + # Tentative start of template argument list + stack.append('<') + elif char in ')]}': + # Found end of parenthesized expression. + # + # If we are currently expecting a matching '>', the pending '<' + # must have been an operator. Remove them from expression stack. + while stack and stack[-1] == '<': + stack.pop() + if not stack: + return (-1, None) + if ((stack[-1] == '(' and char == ')') or + (stack[-1] == '[' and char == ']') or + (stack[-1] == '{' and char == '}')): + stack.pop() + if not stack: + return (i + 1, None) + else: + # Mismatched parentheses + return (-1, None) + elif char == '>': + # Found potential end of template argument list. + + # Ignore "->" and operator functions + if (i > 0 and + (line[i - 1] == '-' or Search(r'\boperator\s*$', line[0:i - 1]))): + continue + + # Pop the stack if there is a matching '<'. Otherwise, ignore + # this '>' since it must be an operator. + if stack: + if stack[-1] == '<': + stack.pop() + if not stack: + return (i + 1, None) + elif char == ';': + # Found something that look like end of statements. If we are currently + # expecting a '>', the matching '<' must have been an operator, since + # template argument list should not contain statements. + while stack and stack[-1] == '<': + stack.pop() + if not stack: + return (-1, None) + + # Did not find end of expression or unbalanced parentheses on this line + return (-1, stack) + + +def CloseExpression(clean_lines, linenum, pos): + """If input points to ( or { or [ or <, finds the position that closes it. + + If lines[linenum][pos] points to a '(' or '{' or '[' or '<', finds the + linenum/pos that correspond to the closing of the expression. + + TODO(unknown): cpplint spends a fair bit of time matching parentheses. + Ideally we would want to index all opening and closing parentheses once + and have CloseExpression be just a simple lookup, but due to preprocessor + tricks, this is not so easy. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + pos: A position on the line. + + Returns: + A tuple (line, linenum, pos) pointer *past* the closing brace, or + (line, len(lines), -1) if we never find a close. Note we ignore + strings and comments when matching; and the line we return is the + 'cleansed' line at linenum. + """ + + line = clean_lines.elided[linenum] + if (line[pos] not in '({[<') or Match(r'<[<=]', line[pos:]): + return (line, clean_lines.NumLines(), -1) + + # Check first line + (end_pos, stack) = FindEndOfExpressionInLine(line, pos, []) + if end_pos > -1: + return (line, linenum, end_pos) + + # Continue scanning forward + while stack and linenum < clean_lines.NumLines() - 1: + linenum += 1 + line = clean_lines.elided[linenum] + (end_pos, stack) = FindEndOfExpressionInLine(line, 0, stack) + if end_pos > -1: + return (line, linenum, end_pos) + + # Did not find end of expression before end of file, give up + return (line, clean_lines.NumLines(), -1) + + +def FindStartOfExpressionInLine(line, endpos, stack): + """Find position at the matching start of current expression. + + This is almost the reverse of FindEndOfExpressionInLine, but note + that the input position and returned position differs by 1. + + Args: + line: a CleansedLines line. + endpos: start searching at this position. + stack: nesting stack at endpos. + + Returns: + On finding matching start: (index at matching start, None) + On finding an unclosed expression: (-1, None) + Otherwise: (-1, new stack at beginning of this line) + """ + i = endpos + while i >= 0: + char = line[i] + if char in ')]}': + # Found end of expression, push to expression stack + stack.append(char) + elif char == '>': + # Found potential end of template argument list. + # + # Ignore it if it's a "->" or ">=" or "operator>" + if (i > 0 and + (line[i - 1] == '-' or + Match(r'\s>=\s', line[i - 1:]) or + Search(r'\boperator\s*$', line[0:i]))): + i -= 1 + else: + stack.append('>') + elif char == '<': + # Found potential start of template argument list + if i > 0 and line[i - 1] == '<': + # Left shift operator + i -= 1 + else: + # If there is a matching '>', we can pop the expression stack. + # Otherwise, ignore this '<' since it must be an operator. + if stack and stack[-1] == '>': + stack.pop() + if not stack: + return (i, None) + elif char in '([{': + # Found start of expression. + # + # If there are any unmatched '>' on the stack, they must be + # operators. Remove those. + while stack and stack[-1] == '>': + stack.pop() + if not stack: + return (-1, None) + if ((char == '(' and stack[-1] == ')') or + (char == '[' and stack[-1] == ']') or + (char == '{' and stack[-1] == '}')): + stack.pop() + if not stack: + return (i, None) + else: + # Mismatched parentheses + return (-1, None) + elif char == ';': + # Found something that look like end of statements. If we are currently + # expecting a '<', the matching '>' must have been an operator, since + # template argument list should not contain statements. + while stack and stack[-1] == '>': + stack.pop() + if not stack: + return (-1, None) + + i -= 1 + + return (-1, stack) + + +def ReverseCloseExpression(clean_lines, linenum, pos): + """If input points to ) or } or ] or >, finds the position that opens it. + + If lines[linenum][pos] points to a ')' or '}' or ']' or '>', finds the + linenum/pos that correspond to the opening of the expression. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + pos: A position on the line. + + Returns: + A tuple (line, linenum, pos) pointer *at* the opening brace, or + (line, 0, -1) if we never find the matching opening brace. Note + we ignore strings and comments when matching; and the line we + return is the 'cleansed' line at linenum. + """ + line = clean_lines.elided[linenum] + if line[pos] not in ')}]>': + return (line, 0, -1) + + # Check last line + (start_pos, stack) = FindStartOfExpressionInLine(line, pos, []) + if start_pos > -1: + return (line, linenum, start_pos) + + # Continue scanning backward + while stack and linenum > 0: + linenum -= 1 + line = clean_lines.elided[linenum] + (start_pos, stack) = FindStartOfExpressionInLine(line, len(line) - 1, stack) + if start_pos > -1: + return (line, linenum, start_pos) + + # Did not find start of expression before beginning of file, give up + return (line, 0, -1) + + +def CheckForCopyright(filename, lines, error): + """Logs an error if no Copyright message appears at the top of the file.""" + + # We'll say it should occur by line 10. Don't forget there's a + # dummy line at the front. + for line in xrange(1, min(len(lines), 11)): + if re.search(r'Copyright', lines[line], re.I): break + else: # means no copyright line was found + error(filename, 0, 'legal/copyright', 5, + 'No copyright message found. ' + 'You should have a line: "Copyright [year] "') + + +def GetIndentLevel(line): + """Return the number of leading spaces in line. + + Args: + line: A string to check. + + Returns: + An integer count of leading spaces, possibly zero. + """ + indent = Match(r'^( *)\S', line) + if indent: + return len(indent.group(1)) + else: + return 0 + +def PathSplitToList(path): + """Returns the path split into a list by the separator. + + Args: + path: An absolute or relative path (e.g. '/a/b/c/' or '../a') + + Returns: + A list of path components (e.g. ['a', 'b', 'c]). + """ + lst = [] + while True: + (head, tail) = os.path.split(path) + if head == path: # absolute paths end + lst.append(head) + break + if tail == path: # relative paths end + lst.append(tail) + break + + path = head + lst.append(tail) + + lst.reverse() + return lst + +def GetHeaderGuardCPPVariable(filename): + """Returns the CPP variable that should be used as a header guard. + + Args: + filename: The name of a C++ header file. + + Returns: + The CPP variable that should be used as a header guard in the + named file. + + """ + + # Restores original filename in case that cpplint is invoked from Emacs's + # flymake. + filename = re.sub(r'_flymake\.h$', '.h', filename) + filename = re.sub(r'/\.flymake/([^/]*)$', r'/\1', filename) + # Replace 'c++' with 'cpp'. + filename = filename.replace('C++', 'cpp').replace('c++', 'cpp') + + fileinfo = FileInfo(filename) + file_path_from_root = fileinfo.RepositoryName() + + def FixupPathFromRoot(): + if _root_debug: + sys.stderr.write("\n_root fixup, _root = '%s', repository name = '%s'\n" + % (_root, fileinfo.RepositoryName())) + + # Process the file path with the --root flag if it was set. + if not _root: + if _root_debug: + sys.stderr.write("_root unspecified\n") + return file_path_from_root + + def StripListPrefix(lst, prefix): + # f(['x', 'y'], ['w, z']) -> None (not a valid prefix) + if lst[:len(prefix)] != prefix: + return None + # f(['a, 'b', 'c', 'd'], ['a', 'b']) -> ['c', 'd'] + return lst[(len(prefix)):] + + # root behavior: + # --root=subdir , lstrips subdir from the header guard + maybe_path = StripListPrefix(PathSplitToList(file_path_from_root), + PathSplitToList(_root)) + + if _root_debug: + sys.stderr.write(("_root lstrip (maybe_path=%s, file_path_from_root=%s," + + " _root=%s)\n") % (maybe_path, file_path_from_root, _root)) + + if maybe_path: + return os.path.join(*maybe_path) + + # --root=.. , will prepend the outer directory to the header guard + full_path = fileinfo.FullName() + root_abspath = os.path.abspath(_root) + + maybe_path = StripListPrefix(PathSplitToList(full_path), + PathSplitToList(root_abspath)) + + if _root_debug: + sys.stderr.write(("_root prepend (maybe_path=%s, full_path=%s, " + + "root_abspath=%s)\n") % (maybe_path, full_path, root_abspath)) + + if maybe_path: + return os.path.join(*maybe_path) + + if _root_debug: + sys.stderr.write("_root ignore, returning %s\n" % (file_path_from_root)) + + # --root=FAKE_DIR is ignored + return file_path_from_root + + file_path_from_root = FixupPathFromRoot() + return re.sub(r'[^a-zA-Z0-9]', '_', file_path_from_root).upper() + '_' + + +def CheckForHeaderGuard(filename, clean_lines, error): + """Checks that the file contains a header guard. + + Logs an error if no #ifndef header guard is present. For other + headers, checks that the full pathname is used. + + Args: + filename: The name of the C++ header file. + clean_lines: A CleansedLines instance containing the file. + error: The function to call with any errors found. + """ + + # Don't check for header guards if there are error suppression + # comments somewhere in this file. + # + # Because this is silencing a warning for a nonexistent line, we + # only support the very specific NOLINT(build/header_guard) syntax, + # and not the general NOLINT or NOLINT(*) syntax. + raw_lines = clean_lines.lines_without_raw_strings + for i in raw_lines: + if Search(r'//\s*NOLINT\(build/header_guard\)', i): + return + + # Allow pragma once instead of header guards + for i in raw_lines: + if Search(r'^\s*#pragma\s+once', i): + return + + cppvar = GetHeaderGuardCPPVariable(filename) + + ifndef = '' + ifndef_linenum = 0 + define = '' + endif = '' + endif_linenum = 0 + for linenum, line in enumerate(raw_lines): + linesplit = line.split() + if len(linesplit) >= 2: + # find the first occurrence of #ifndef and #define, save arg + if not ifndef and linesplit[0] == '#ifndef': + # set ifndef to the header guard presented on the #ifndef line. + ifndef = linesplit[1] + ifndef_linenum = linenum + if not define and linesplit[0] == '#define': + define = linesplit[1] + # find the last occurrence of #endif, save entire line + if line.startswith('#endif'): + endif = line + endif_linenum = linenum + + if not ifndef or not define or ifndef != define: + error(filename, 0, 'build/header_guard', 5, + 'No #ifndef header guard found, suggested CPP variable is: %s' % + cppvar) + return + + # The guard should be PATH_FILE_H_, but we also allow PATH_FILE_H__ + # for backward compatibility. + if ifndef != cppvar: + error_level = 0 + if ifndef != cppvar + '_': + error_level = 5 + + ParseNolintSuppressions(filename, raw_lines[ifndef_linenum], ifndef_linenum, + error) + error(filename, ifndef_linenum, 'build/header_guard', error_level, + '#ifndef header guard has wrong style, please use: %s' % cppvar) + + # Check for "//" comments on endif line. + ParseNolintSuppressions(filename, raw_lines[endif_linenum], endif_linenum, + error) + match = Match(r'#endif\s*//\s*' + cppvar + r'(_)?\b', endif) + if match: + if match.group(1) == '_': + # Issue low severity warning for deprecated double trailing underscore + error(filename, endif_linenum, 'build/header_guard', 0, + '#endif line should be "#endif // %s"' % cppvar) + return + + # Didn't find the corresponding "//" comment. If this file does not + # contain any "//" comments at all, it could be that the compiler + # only wants "/**/" comments, look for those instead. + no_single_line_comments = True + for i in xrange(1, len(raw_lines) - 1): + line = raw_lines[i] + if Match(r'^(?:(?:\'(?:\.|[^\'])*\')|(?:"(?:\.|[^"])*")|[^\'"])*//', line): + no_single_line_comments = False + break + + if no_single_line_comments: + match = Match(r'#endif\s*/\*\s*' + cppvar + r'(_)?\s*\*/', endif) + if match: + if match.group(1) == '_': + # Low severity warning for double trailing underscore + error(filename, endif_linenum, 'build/header_guard', 0, + '#endif line should be "#endif /* %s */"' % cppvar) + return + + # Didn't find anything + error(filename, endif_linenum, 'build/header_guard', 5, + '#endif line should be "#endif // %s"' % cppvar) + + +def CheckHeaderFileIncluded(filename, include_state, error): + """Logs an error if a source file does not include its header.""" + + # Do not check test files + fileinfo = FileInfo(filename) + if Search(_TEST_FILE_SUFFIX, fileinfo.BaseName()): + return + + for ext in GetHeaderExtensions(): + basefilename = filename[0:len(filename) - len(fileinfo.Extension())] + headerfile = basefilename + '.' + ext + if not os.path.exists(headerfile): + continue + headername = FileInfo(headerfile).RepositoryName() + first_include = None + for section_list in include_state.include_list: + for f in section_list: + if headername in f[0] or f[0] in headername: + return + if not first_include: + first_include = f[1] + + error(filename, first_include, 'build/include', 5, + '%s should include its header file %s' % (fileinfo.RepositoryName(), + headername)) + + +def CheckForBadCharacters(filename, lines, error): + """Logs an error for each line containing bad characters. + + Two kinds of bad characters: + + 1. Unicode replacement characters: These indicate that either the file + contained invalid UTF-8 (likely) or Unicode replacement characters (which + it shouldn't). Note that it's possible for this to throw off line + numbering if the invalid UTF-8 occurred adjacent to a newline. + + 2. NUL bytes. These are problematic for some tools. + + Args: + filename: The name of the current file. + lines: An array of strings, each representing a line of the file. + error: The function to call with any errors found. + """ + for linenum, line in enumerate(lines): + if unicode_escape_decode('\ufffd') in line: + error(filename, linenum, 'readability/utf8', 5, + 'Line contains invalid UTF-8 (or Unicode replacement character).') + if '\0' in line: + error(filename, linenum, 'readability/nul', 5, 'Line contains NUL byte.') + + +def CheckForNewlineAtEOF(filename, lines, error): + """Logs an error if there is no newline char at the end of the file. + + Args: + filename: The name of the current file. + lines: An array of strings, each representing a line of the file. + error: The function to call with any errors found. + """ + + # The array lines() was created by adding two newlines to the + # original file (go figure), then splitting on \n. + # To verify that the file ends in \n, we just have to make sure the + # last-but-two element of lines() exists and is empty. + if len(lines) < 3 or lines[-2]: + error(filename, len(lines) - 2, 'whitespace/ending_newline', 5, + 'Could not find a newline character at the end of the file.') + + +def CheckForMultilineCommentsAndStrings(filename, clean_lines, linenum, error): + """Logs an error if we see /* ... */ or "..." that extend past one line. + + /* ... */ comments are legit inside macros, for one line. + Otherwise, we prefer // comments, so it's ok to warn about the + other. Likewise, it's ok for strings to extend across multiple + lines, as long as a line continuation character (backslash) + terminates each line. Although not currently prohibited by the C++ + style guide, it's ugly and unnecessary. We don't do well with either + in this lint program, so we warn about both. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Remove all \\ (escaped backslashes) from the line. They are OK, and the + # second (escaped) slash may trigger later \" detection erroneously. + line = line.replace('\\\\', '') + + if line.count('/*') > line.count('*/'): + error(filename, linenum, 'readability/multiline_comment', 5, + 'Complex multi-line /*...*/-style comment found. ' + 'Lint may give bogus warnings. ' + 'Consider replacing these with //-style comments, ' + 'with #if 0...#endif, ' + 'or with more clearly structured multi-line comments.') + + if (line.count('"') - line.count('\\"')) % 2: + error(filename, linenum, 'readability/multiline_string', 5, + 'Multi-line string ("...") found. This lint script doesn\'t ' + 'do well with such strings, and may give bogus warnings. ' + 'Use C++11 raw strings or concatenation instead.') + + +# (non-threadsafe name, thread-safe alternative, validation pattern) +# +# The validation pattern is used to eliminate false positives such as: +# _rand(); // false positive due to substring match. +# ->rand(); // some member function rand(). +# ACMRandom rand(seed); // some variable named rand. +# ISAACRandom rand(); // another variable named rand. +# +# Basically we require the return value of these functions to be used +# in some expression context on the same line by matching on some +# operator before the function name. This eliminates constructors and +# member function calls. +_UNSAFE_FUNC_PREFIX = r'(?:[-+*/=%^&|(<]\s*|>\s+)' +_THREADING_LIST = ( + ('asctime(', 'asctime_r(', _UNSAFE_FUNC_PREFIX + r'asctime\([^)]+\)'), + ('ctime(', 'ctime_r(', _UNSAFE_FUNC_PREFIX + r'ctime\([^)]+\)'), + ('getgrgid(', 'getgrgid_r(', _UNSAFE_FUNC_PREFIX + r'getgrgid\([^)]+\)'), + ('getgrnam(', 'getgrnam_r(', _UNSAFE_FUNC_PREFIX + r'getgrnam\([^)]+\)'), + ('getlogin(', 'getlogin_r(', _UNSAFE_FUNC_PREFIX + r'getlogin\(\)'), + ('getpwnam(', 'getpwnam_r(', _UNSAFE_FUNC_PREFIX + r'getpwnam\([^)]+\)'), + ('getpwuid(', 'getpwuid_r(', _UNSAFE_FUNC_PREFIX + r'getpwuid\([^)]+\)'), + ('gmtime(', 'gmtime_r(', _UNSAFE_FUNC_PREFIX + r'gmtime\([^)]+\)'), + ('localtime(', 'localtime_r(', _UNSAFE_FUNC_PREFIX + r'localtime\([^)]+\)'), + ('rand(', 'rand_r(', _UNSAFE_FUNC_PREFIX + r'rand\(\)'), + ('strtok(', 'strtok_r(', + _UNSAFE_FUNC_PREFIX + r'strtok\([^)]+\)'), + ('ttyname(', 'ttyname_r(', _UNSAFE_FUNC_PREFIX + r'ttyname\([^)]+\)'), + ) + + +def CheckPosixThreading(filename, clean_lines, linenum, error): + """Checks for calls to thread-unsafe functions. + + Much code has been originally written without consideration of + multi-threading. Also, engineers are relying on their old experience; + they have learned posix before threading extensions were added. These + tests guide the engineers to use thread-safe functions (when using + posix directly). + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + for single_thread_func, multithread_safe_func, pattern in _THREADING_LIST: + # Additional pattern matching check to confirm that this is the + # function we are looking for + if Search(pattern, line): + error(filename, linenum, 'runtime/threadsafe_fn', 2, + 'Consider using ' + multithread_safe_func + + '...) instead of ' + single_thread_func + + '...) for improved thread safety.') + + +def CheckVlogArguments(filename, clean_lines, linenum, error): + """Checks that VLOG() is only used for defining a logging level. + + For example, VLOG(2) is correct. VLOG(INFO), VLOG(WARNING), VLOG(ERROR), and + VLOG(FATAL) are not. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + if Search(r'\bVLOG\((INFO|ERROR|WARNING|DFATAL|FATAL)\)', line): + error(filename, linenum, 'runtime/vlog', 5, + 'VLOG() should be used with numeric verbosity level. ' + 'Use LOG() if you want symbolic severity levels.') + +# Matches invalid increment: *count++, which moves pointer instead of +# incrementing a value. +_RE_PATTERN_INVALID_INCREMENT = re.compile( + r'^\s*\*\w+(\+\+|--);') + + +def CheckInvalidIncrement(filename, clean_lines, linenum, error): + """Checks for invalid increment *count++. + + For example following function: + void increment_counter(int* count) { + *count++; + } + is invalid, because it effectively does count++, moving pointer, and should + be replaced with ++*count, (*count)++ or *count += 1. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + if _RE_PATTERN_INVALID_INCREMENT.match(line): + error(filename, linenum, 'runtime/invalid_increment', 5, + 'Changing pointer instead of value (or unused value of operator*).') + + +def IsMacroDefinition(clean_lines, linenum): + if Search(r'^#define', clean_lines[linenum]): + return True + + if linenum > 0 and Search(r'\\$', clean_lines[linenum - 1]): + return True + + return False + + +def IsForwardClassDeclaration(clean_lines, linenum): + return Match(r'^\s*(\btemplate\b)*.*class\s+\w+;\s*$', clean_lines[linenum]) + + +class _BlockInfo(object): + """Stores information about a generic block of code.""" + + def __init__(self, linenum, seen_open_brace): + self.starting_linenum = linenum + self.seen_open_brace = seen_open_brace + self.open_parentheses = 0 + self.inline_asm = _NO_ASM + self.check_namespace_indentation = False + + def CheckBegin(self, filename, clean_lines, linenum, error): + """Run checks that applies to text up to the opening brace. + + This is mostly for checking the text after the class identifier + and the "{", usually where the base class is specified. For other + blocks, there isn't much to check, so we always pass. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + pass + + def CheckEnd(self, filename, clean_lines, linenum, error): + """Run checks that applies to text after the closing brace. + + This is mostly used for checking end of namespace comments. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + pass + + def IsBlockInfo(self): + """Returns true if this block is a _BlockInfo. + + This is convenient for verifying that an object is an instance of + a _BlockInfo, but not an instance of any of the derived classes. + + Returns: + True for this class, False for derived classes. + """ + return self.__class__ == _BlockInfo + + +class _ExternCInfo(_BlockInfo): + """Stores information about an 'extern "C"' block.""" + + def __init__(self, linenum): + _BlockInfo.__init__(self, linenum, True) + + +class _ClassInfo(_BlockInfo): + """Stores information about a class.""" + + def __init__(self, name, class_or_struct, clean_lines, linenum): + _BlockInfo.__init__(self, linenum, False) + self.name = name + self.is_derived = False + self.check_namespace_indentation = True + if class_or_struct == 'struct': + self.access = 'public' + self.is_struct = True + else: + self.access = 'private' + self.is_struct = False + + # Remember initial indentation level for this class. Using raw_lines here + # instead of elided to account for leading comments. + self.class_indent = GetIndentLevel(clean_lines.raw_lines[linenum]) + + # Try to find the end of the class. This will be confused by things like: + # class A { + # } *x = { ... + # + # But it's still good enough for CheckSectionSpacing. + self.last_line = 0 + depth = 0 + for i in range(linenum, clean_lines.NumLines()): + line = clean_lines.elided[i] + depth += line.count('{') - line.count('}') + if not depth: + self.last_line = i + break + + def CheckBegin(self, filename, clean_lines, linenum, error): + # Look for a bare ':' + if Search('(^|[^:]):($|[^:])', clean_lines.elided[linenum]): + self.is_derived = True + + def CheckEnd(self, filename, clean_lines, linenum, error): + # If there is a DISALLOW macro, it should appear near the end of + # the class. + seen_last_thing_in_class = False + for i in xrange(linenum - 1, self.starting_linenum, -1): + match = Search( + r'\b(DISALLOW_COPY_AND_ASSIGN|DISALLOW_IMPLICIT_CONSTRUCTORS)\(' + + self.name + r'\)', + clean_lines.elided[i]) + if match: + if seen_last_thing_in_class: + error(filename, i, 'readability/constructors', 3, + match.group(1) + ' should be the last thing in the class') + break + + if not Match(r'^\s*$', clean_lines.elided[i]): + seen_last_thing_in_class = True + + # Check that closing brace is aligned with beginning of the class. + # Only do this if the closing brace is indented by only whitespaces. + # This means we will not check single-line class definitions. + indent = Match(r'^( *)\}', clean_lines.elided[linenum]) + if indent and len(indent.group(1)) != self.class_indent: + if self.is_struct: + parent = 'struct ' + self.name + else: + parent = 'class ' + self.name + error(filename, linenum, 'whitespace/indent', 3, + 'Closing brace should be aligned with beginning of %s' % parent) + + +class _NamespaceInfo(_BlockInfo): + """Stores information about a namespace.""" + + def __init__(self, name, linenum): + _BlockInfo.__init__(self, linenum, False) + self.name = name or '' + self.check_namespace_indentation = True + + def CheckEnd(self, filename, clean_lines, linenum, error): + """Check end of namespace comments.""" + line = clean_lines.raw_lines[linenum] + + # Check how many lines is enclosed in this namespace. Don't issue + # warning for missing namespace comments if there aren't enough + # lines. However, do apply checks if there is already an end of + # namespace comment and it's incorrect. + # + # TODO(unknown): We always want to check end of namespace comments + # if a namespace is large, but sometimes we also want to apply the + # check if a short namespace contained nontrivial things (something + # other than forward declarations). There is currently no logic on + # deciding what these nontrivial things are, so this check is + # triggered by namespace size only, which works most of the time. + if (linenum - self.starting_linenum < 10 + and not Match(r'^\s*};*\s*(//|/\*).*\bnamespace\b', line)): + return + + # Look for matching comment at end of namespace. + # + # Note that we accept C style "/* */" comments for terminating + # namespaces, so that code that terminate namespaces inside + # preprocessor macros can be cpplint clean. + # + # We also accept stuff like "// end of namespace ." with the + # period at the end. + # + # Besides these, we don't accept anything else, otherwise we might + # get false negatives when existing comment is a substring of the + # expected namespace. + if self.name: + # Named namespace + if not Match((r'^\s*};*\s*(//|/\*).*\bnamespace\s+' + + re.escape(self.name) + r'[\*/\.\\\s]*$'), + line): + error(filename, linenum, 'readability/namespace', 5, + 'Namespace should be terminated with "// namespace %s"' % + self.name) + else: + # Anonymous namespace + if not Match(r'^\s*};*\s*(//|/\*).*\bnamespace[\*/\.\\\s]*$', line): + # If "// namespace anonymous" or "// anonymous namespace (more text)", + # mention "// anonymous namespace" as an acceptable form + if Match(r'^\s*}.*\b(namespace anonymous|anonymous namespace)\b', line): + error(filename, linenum, 'readability/namespace', 5, + 'Anonymous namespace should be terminated with "// namespace"' + ' or "// anonymous namespace"') + else: + error(filename, linenum, 'readability/namespace', 5, + 'Anonymous namespace should be terminated with "// namespace"') + + +class _PreprocessorInfo(object): + """Stores checkpoints of nesting stacks when #if/#else is seen.""" + + def __init__(self, stack_before_if): + # The entire nesting stack before #if + self.stack_before_if = stack_before_if + + # The entire nesting stack up to #else + self.stack_before_else = [] + + # Whether we have already seen #else or #elif + self.seen_else = False + + +class NestingState(object): + """Holds states related to parsing braces.""" + + def __init__(self): + # Stack for tracking all braces. An object is pushed whenever we + # see a "{", and popped when we see a "}". Only 3 types of + # objects are possible: + # - _ClassInfo: a class or struct. + # - _NamespaceInfo: a namespace. + # - _BlockInfo: some other type of block. + self.stack = [] + + # Top of the previous stack before each Update(). + # + # Because the nesting_stack is updated at the end of each line, we + # had to do some convoluted checks to find out what is the current + # scope at the beginning of the line. This check is simplified by + # saving the previous top of nesting stack. + # + # We could save the full stack, but we only need the top. Copying + # the full nesting stack would slow down cpplint by ~10%. + self.previous_stack_top = [] + + # Stack of _PreprocessorInfo objects. + self.pp_stack = [] + + def SeenOpenBrace(self): + """Check if we have seen the opening brace for the innermost block. + + Returns: + True if we have seen the opening brace, False if the innermost + block is still expecting an opening brace. + """ + return (not self.stack) or self.stack[-1].seen_open_brace + + def InNamespaceBody(self): + """Check if we are currently one level inside a namespace body. + + Returns: + True if top of the stack is a namespace block, False otherwise. + """ + return self.stack and isinstance(self.stack[-1], _NamespaceInfo) + + def InExternC(self): + """Check if we are currently one level inside an 'extern "C"' block. + + Returns: + True if top of the stack is an extern block, False otherwise. + """ + return self.stack and isinstance(self.stack[-1], _ExternCInfo) + + def InClassDeclaration(self): + """Check if we are currently one level inside a class or struct declaration. + + Returns: + True if top of the stack is a class/struct, False otherwise. + """ + return self.stack and isinstance(self.stack[-1], _ClassInfo) + + def InAsmBlock(self): + """Check if we are currently one level inside an inline ASM block. + + Returns: + True if the top of the stack is a block containing inline ASM. + """ + return self.stack and self.stack[-1].inline_asm != _NO_ASM + + def InTemplateArgumentList(self, clean_lines, linenum, pos): + """Check if current position is inside template argument list. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + pos: position just after the suspected template argument. + Returns: + True if (linenum, pos) is inside template arguments. + """ + while linenum < clean_lines.NumLines(): + # Find the earliest character that might indicate a template argument + line = clean_lines.elided[linenum] + match = Match(r'^[^{};=\[\]\.<>]*(.)', line[pos:]) + if not match: + linenum += 1 + pos = 0 + continue + token = match.group(1) + pos += len(match.group(0)) + + # These things do not look like template argument list: + # class Suspect { + # class Suspect x; } + if token in ('{', '}', ';'): return False + + # These things look like template argument list: + # template + # template + # template + # template + if token in ('>', '=', '[', ']', '.'): return True + + # Check if token is an unmatched '<'. + # If not, move on to the next character. + if token != '<': + pos += 1 + if pos >= len(line): + linenum += 1 + pos = 0 + continue + + # We can't be sure if we just find a single '<', and need to + # find the matching '>'. + (_, end_line, end_pos) = CloseExpression(clean_lines, linenum, pos - 1) + if end_pos < 0: + # Not sure if template argument list or syntax error in file + return False + linenum = end_line + pos = end_pos + return False + + def UpdatePreprocessor(self, line): + """Update preprocessor stack. + + We need to handle preprocessors due to classes like this: + #ifdef SWIG + struct ResultDetailsPageElementExtensionPoint { + #else + struct ResultDetailsPageElementExtensionPoint : public Extension { + #endif + + We make the following assumptions (good enough for most files): + - Preprocessor condition evaluates to true from #if up to first + #else/#elif/#endif. + + - Preprocessor condition evaluates to false from #else/#elif up + to #endif. We still perform lint checks on these lines, but + these do not affect nesting stack. + + Args: + line: current line to check. + """ + if Match(r'^\s*#\s*(if|ifdef|ifndef)\b', line): + # Beginning of #if block, save the nesting stack here. The saved + # stack will allow us to restore the parsing state in the #else case. + self.pp_stack.append(_PreprocessorInfo(copy.deepcopy(self.stack))) + elif Match(r'^\s*#\s*(else|elif)\b', line): + # Beginning of #else block + if self.pp_stack: + if not self.pp_stack[-1].seen_else: + # This is the first #else or #elif block. Remember the + # whole nesting stack up to this point. This is what we + # keep after the #endif. + self.pp_stack[-1].seen_else = True + self.pp_stack[-1].stack_before_else = copy.deepcopy(self.stack) + + # Restore the stack to how it was before the #if + self.stack = copy.deepcopy(self.pp_stack[-1].stack_before_if) + else: + # TODO(unknown): unexpected #else, issue warning? + pass + elif Match(r'^\s*#\s*endif\b', line): + # End of #if or #else blocks. + if self.pp_stack: + # If we saw an #else, we will need to restore the nesting + # stack to its former state before the #else, otherwise we + # will just continue from where we left off. + if self.pp_stack[-1].seen_else: + # Here we can just use a shallow copy since we are the last + # reference to it. + self.stack = self.pp_stack[-1].stack_before_else + # Drop the corresponding #if + self.pp_stack.pop() + else: + # TODO(unknown): unexpected #endif, issue warning? + pass + + # TODO(unknown): Update() is too long, but we will refactor later. + def Update(self, filename, clean_lines, linenum, error): + """Update nesting state with current line. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Remember top of the previous nesting stack. + # + # The stack is always pushed/popped and not modified in place, so + # we can just do a shallow copy instead of copy.deepcopy. Using + # deepcopy would slow down cpplint by ~28%. + if self.stack: + self.previous_stack_top = self.stack[-1] + else: + self.previous_stack_top = None + + # Update pp_stack + self.UpdatePreprocessor(line) + + # Count parentheses. This is to avoid adding struct arguments to + # the nesting stack. + if self.stack: + inner_block = self.stack[-1] + depth_change = line.count('(') - line.count(')') + inner_block.open_parentheses += depth_change + + # Also check if we are starting or ending an inline assembly block. + if inner_block.inline_asm in (_NO_ASM, _END_ASM): + if (depth_change != 0 and + inner_block.open_parentheses == 1 and + _MATCH_ASM.match(line)): + # Enter assembly block + inner_block.inline_asm = _INSIDE_ASM + else: + # Not entering assembly block. If previous line was _END_ASM, + # we will now shift to _NO_ASM state. + inner_block.inline_asm = _NO_ASM + elif (inner_block.inline_asm == _INSIDE_ASM and + inner_block.open_parentheses == 0): + # Exit assembly block + inner_block.inline_asm = _END_ASM + + # Consume namespace declaration at the beginning of the line. Do + # this in a loop so that we catch same line declarations like this: + # namespace proto2 { namespace bridge { class MessageSet; } } + while True: + # Match start of namespace. The "\b\s*" below catches namespace + # declarations even if it weren't followed by a whitespace, this + # is so that we don't confuse our namespace checker. The + # missing spaces will be flagged by CheckSpacing. + namespace_decl_match = Match(r'^\s*namespace\b\s*([:\w]+)?(.*)$', line) + if not namespace_decl_match: + break + + new_namespace = _NamespaceInfo(namespace_decl_match.group(1), linenum) + self.stack.append(new_namespace) + + line = namespace_decl_match.group(2) + if line.find('{') != -1: + new_namespace.seen_open_brace = True + line = line[line.find('{') + 1:] + + # Look for a class declaration in whatever is left of the line + # after parsing namespaces. The regexp accounts for decorated classes + # such as in: + # class LOCKABLE API Object { + # }; + class_decl_match = Match( + r'^(\s*(?:template\s*<[\w\s<>,:=]*>\s*)?' + r'(class|struct)\s+(?:[A-Z_]+\s+)*(\w+(?:::\w+)*))' + r'(.*)$', line) + if (class_decl_match and + (not self.stack or self.stack[-1].open_parentheses == 0)): + # We do not want to accept classes that are actually template arguments: + # template , + # template class Ignore3> + # void Function() {}; + # + # To avoid template argument cases, we scan forward and look for + # an unmatched '>'. If we see one, assume we are inside a + # template argument list. + end_declaration = len(class_decl_match.group(1)) + if not self.InTemplateArgumentList(clean_lines, linenum, end_declaration): + self.stack.append(_ClassInfo( + class_decl_match.group(3), class_decl_match.group(2), + clean_lines, linenum)) + line = class_decl_match.group(4) + + # If we have not yet seen the opening brace for the innermost block, + # run checks here. + if not self.SeenOpenBrace(): + self.stack[-1].CheckBegin(filename, clean_lines, linenum, error) + + # Update access control if we are inside a class/struct + if self.stack and isinstance(self.stack[-1], _ClassInfo): + classinfo = self.stack[-1] + access_match = Match( + r'^(.*)\b(public|private|protected|signals)(\s+(?:slots\s*)?)?' + r':(?:[^:]|$)', + line) + if access_match: + classinfo.access = access_match.group(2) + + # Check that access keywords are indented +1 space. Skip this + # check if the keywords are not preceded by whitespaces. + indent = access_match.group(1) + if (len(indent) != classinfo.class_indent + 1 and + Match(r'^\s*$', indent)): + if classinfo.is_struct: + parent = 'struct ' + classinfo.name + else: + parent = 'class ' + classinfo.name + slots = '' + if access_match.group(3): + slots = access_match.group(3) + error(filename, linenum, 'whitespace/indent', 3, + '%s%s: should be indented +1 space inside %s' % ( + access_match.group(2), slots, parent)) + + # Consume braces or semicolons from what's left of the line + while True: + # Match first brace, semicolon, or closed parenthesis. + matched = Match(r'^[^{;)}]*([{;)}])(.*)$', line) + if not matched: + break + + token = matched.group(1) + if token == '{': + # If namespace or class hasn't seen a opening brace yet, mark + # namespace/class head as complete. Push a new block onto the + # stack otherwise. + if not self.SeenOpenBrace(): + self.stack[-1].seen_open_brace = True + elif Match(r'^extern\s*"[^"]*"\s*\{', line): + self.stack.append(_ExternCInfo(linenum)) + else: + self.stack.append(_BlockInfo(linenum, True)) + if _MATCH_ASM.match(line): + self.stack[-1].inline_asm = _BLOCK_ASM + + elif token == ';' or token == ')': + # If we haven't seen an opening brace yet, but we already saw + # a semicolon, this is probably a forward declaration. Pop + # the stack for these. + # + # Similarly, if we haven't seen an opening brace yet, but we + # already saw a closing parenthesis, then these are probably + # function arguments with extra "class" or "struct" keywords. + # Also pop these stack for these. + if not self.SeenOpenBrace(): + self.stack.pop() + else: # token == '}' + # Perform end of block checks and pop the stack. + if self.stack: + self.stack[-1].CheckEnd(filename, clean_lines, linenum, error) + self.stack.pop() + line = matched.group(2) + + def InnermostClass(self): + """Get class info on the top of the stack. + + Returns: + A _ClassInfo object if we are inside a class, or None otherwise. + """ + for i in range(len(self.stack), 0, -1): + classinfo = self.stack[i - 1] + if isinstance(classinfo, _ClassInfo): + return classinfo + return None + + def CheckCompletedBlocks(self, filename, error): + """Checks that all classes and namespaces have been completely parsed. + + Call this when all lines in a file have been processed. + Args: + filename: The name of the current file. + error: The function to call with any errors found. + """ + # Note: This test can result in false positives if #ifdef constructs + # get in the way of brace matching. See the testBuildClass test in + # cpplint_unittest.py for an example of this. + for obj in self.stack: + if isinstance(obj, _ClassInfo): + error(filename, obj.starting_linenum, 'build/class', 5, + 'Failed to find complete declaration of class %s' % + obj.name) + elif isinstance(obj, _NamespaceInfo): + error(filename, obj.starting_linenum, 'build/namespaces', 5, + 'Failed to find complete declaration of namespace %s' % + obj.name) + + +def CheckForNonStandardConstructs(filename, clean_lines, linenum, + nesting_state, error): + r"""Logs an error if we see certain non-ANSI constructs ignored by gcc-2. + + Complain about several constructs which gcc-2 accepts, but which are + not standard C++. Warning about these in lint is one way to ease the + transition to new compilers. + - put storage class first (e.g. "static const" instead of "const static"). + - "%lld" instead of %qd" in printf-type functions. + - "%1$d" is non-standard in printf-type functions. + - "\%" is an undefined character escape sequence. + - text after #endif is not allowed. + - invalid inner-style forward declaration. + - >? and ?= and )\?=?\s*(\w+|[+-]?\d+)(\.\d*)?', + line): + error(filename, linenum, 'build/deprecated', 3, + '>? and ))?' + # r'\s*const\s*' + type_name + '\s*&\s*\w+\s*;' + error(filename, linenum, 'runtime/member_string_references', 2, + 'const string& members are dangerous. It is much better to use ' + 'alternatives, such as pointers or simple constants.') + + # Everything else in this function operates on class declarations. + # Return early if the top of the nesting stack is not a class, or if + # the class head is not completed yet. + classinfo = nesting_state.InnermostClass() + if not classinfo or not classinfo.seen_open_brace: + return + + # The class may have been declared with namespace or classname qualifiers. + # The constructor and destructor will not have those qualifiers. + base_classname = classinfo.name.split('::')[-1] + + # Look for single-argument constructors that aren't marked explicit. + # Technically a valid construct, but against style. + explicit_constructor_match = Match( + r'\s+(?:(?:inline|constexpr)\s+)*(explicit\s+)?' + r'(?:(?:inline|constexpr)\s+)*%s\s*' + r'\(((?:[^()]|\([^()]*\))*)\)' + % re.escape(base_classname), + line) + + if explicit_constructor_match: + is_marked_explicit = explicit_constructor_match.group(1) + + if not explicit_constructor_match.group(2): + constructor_args = [] + else: + constructor_args = explicit_constructor_match.group(2).split(',') + + # collapse arguments so that commas in template parameter lists and function + # argument parameter lists don't split arguments in two + i = 0 + while i < len(constructor_args): + constructor_arg = constructor_args[i] + while (constructor_arg.count('<') > constructor_arg.count('>') or + constructor_arg.count('(') > constructor_arg.count(')')): + constructor_arg += ',' + constructor_args[i + 1] + del constructor_args[i + 1] + constructor_args[i] = constructor_arg + i += 1 + + variadic_args = [arg for arg in constructor_args if '&&...' in arg] + defaulted_args = [arg for arg in constructor_args if '=' in arg] + noarg_constructor = (not constructor_args or # empty arg list + # 'void' arg specifier + (len(constructor_args) == 1 and + constructor_args[0].strip() == 'void')) + onearg_constructor = ((len(constructor_args) == 1 and # exactly one arg + not noarg_constructor) or + # all but at most one arg defaulted + (len(constructor_args) >= 1 and + not noarg_constructor and + len(defaulted_args) >= len(constructor_args) - 1) or + # variadic arguments with zero or one argument + (len(constructor_args) <= 2 and + len(variadic_args) >= 1)) + initializer_list_constructor = bool( + onearg_constructor and + Search(r'\bstd\s*::\s*initializer_list\b', constructor_args[0])) + copy_constructor = bool( + onearg_constructor and + Match(r'(const\s+)?%s(\s*<[^>]*>)?(\s+const)?\s*(?:<\w+>\s*)?&' + % re.escape(base_classname), constructor_args[0].strip())) + + if (not is_marked_explicit and + onearg_constructor and + not initializer_list_constructor and + not copy_constructor): + if defaulted_args or variadic_args: + error(filename, linenum, 'runtime/explicit', 5, + 'Constructors callable with one argument ' + 'should be marked explicit.') + else: + error(filename, linenum, 'runtime/explicit', 5, + 'Single-parameter constructors should be marked explicit.') + elif is_marked_explicit and not onearg_constructor: + if noarg_constructor: + error(filename, linenum, 'runtime/explicit', 5, + 'Zero-parameter constructors should not be marked explicit.') + + +def CheckSpacingForFunctionCall(filename, clean_lines, linenum, error): + """Checks for the correctness of various spacing around function calls. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Since function calls often occur inside if/for/while/switch + # expressions - which have their own, more liberal conventions - we + # first see if we should be looking inside such an expression for a + # function call, to which we can apply more strict standards. + fncall = line # if there's no control flow construct, look at whole line + for pattern in (r'\bif\s*\((.*)\)\s*{', + r'\bfor\s*\((.*)\)\s*{', + r'\bwhile\s*\((.*)\)\s*[{;]', + r'\bswitch\s*\((.*)\)\s*{'): + match = Search(pattern, line) + if match: + fncall = match.group(1) # look inside the parens for function calls + break + + # Except in if/for/while/switch, there should never be space + # immediately inside parens (eg "f( 3, 4 )"). We make an exception + # for nested parens ( (a+b) + c ). Likewise, there should never be + # a space before a ( when it's a function argument. I assume it's a + # function argument when the char before the whitespace is legal in + # a function name (alnum + _) and we're not starting a macro. Also ignore + # pointers and references to arrays and functions coz they're too tricky: + # we use a very simple way to recognize these: + # " (something)(maybe-something)" or + # " (something)(maybe-something," or + # " (something)[something]" + # Note that we assume the contents of [] to be short enough that + # they'll never need to wrap. + if ( # Ignore control structures. + not Search(r'\b(if|for|while|switch|return|new|delete|catch|sizeof)\b', + fncall) and + # Ignore pointers/references to functions. + not Search(r' \([^)]+\)\([^)]*(\)|,$)', fncall) and + # Ignore pointers/references to arrays. + not Search(r' \([^)]+\)\[[^\]]+\]', fncall)): + if Search(r'\w\s*\(\s(?!\s*\\$)', fncall): # a ( used for a fn call + error(filename, linenum, 'whitespace/parens', 4, + 'Extra space after ( in function call') + elif Search(r'\(\s+(?!(\s*\\)|\()', fncall): + error(filename, linenum, 'whitespace/parens', 2, + 'Extra space after (') + if (Search(r'\w\s+\(', fncall) and + not Search(r'_{0,2}asm_{0,2}\s+_{0,2}volatile_{0,2}\s+\(', fncall) and + not Search(r'#\s*define|typedef|using\s+\w+\s*=', fncall) and + not Search(r'\w\s+\((\w+::)*\*\w+\)\(', fncall) and + not Search(r'\bcase\s+\(', fncall)): + # TODO(unknown): Space after an operator function seem to be a common + # error, silence those for now by restricting them to highest verbosity. + if Search(r'\boperator_*\b', line): + error(filename, linenum, 'whitespace/parens', 0, + 'Extra space before ( in function call') + else: + error(filename, linenum, 'whitespace/parens', 4, + 'Extra space before ( in function call') + # If the ) is followed only by a newline or a { + newline, assume it's + # part of a control statement (if/while/etc), and don't complain + if Search(r'[^)]\s+\)\s*[^{\s]', fncall): + # If the closing parenthesis is preceded by only whitespaces, + # try to give a more descriptive error message. + if Search(r'^\s+\)', fncall): + error(filename, linenum, 'whitespace/parens', 2, + 'Closing ) should be moved to the previous line') + else: + error(filename, linenum, 'whitespace/parens', 2, + 'Extra space before )') + + +def IsBlankLine(line): + """Returns true if the given line is blank. + + We consider a line to be blank if the line is empty or consists of + only white spaces. + + Args: + line: A line of a string. + + Returns: + True, if the given line is blank. + """ + return not line or line.isspace() + + +def CheckForNamespaceIndentation(filename, nesting_state, clean_lines, line, + error): + is_namespace_indent_item = ( + len(nesting_state.stack) > 1 and + nesting_state.stack[-1].check_namespace_indentation and + isinstance(nesting_state.previous_stack_top, _NamespaceInfo) and + nesting_state.previous_stack_top == nesting_state.stack[-2]) + + if ShouldCheckNamespaceIndentation(nesting_state, is_namespace_indent_item, + clean_lines.elided, line): + CheckItemIndentationInNamespace(filename, clean_lines.elided, + line, error) + + +def CheckForFunctionLengths(filename, clean_lines, linenum, + function_state, error): + """Reports for long function bodies. + + For an overview why this is done, see: + https://google-styleguide.googlecode.com/svn/trunk/cppguide.xml#Write_Short_Functions + + Uses a simplistic algorithm assuming other style guidelines + (especially spacing) are followed. + Only checks unindented functions, so class members are unchecked. + Trivial bodies are unchecked, so constructors with huge initializer lists + may be missed. + Blank/comment lines are not counted so as to avoid encouraging the removal + of vertical space and comments just to get through a lint check. + NOLINT *on the last line of a function* disables this check. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + function_state: Current function name and lines in body so far. + error: The function to call with any errors found. + """ + lines = clean_lines.lines + line = lines[linenum] + joined_line = '' + + starting_func = False + regexp = r'(\w(\w|::|\*|\&|\s)*)\(' # decls * & space::name( ... + match_result = Match(regexp, line) + if match_result: + # If the name is all caps and underscores, figure it's a macro and + # ignore it, unless it's TEST or TEST_F. + function_name = match_result.group(1).split()[-1] + if function_name == 'TEST' or function_name == 'TEST_F' or ( + not Match(r'[A-Z_]+$', function_name)): + starting_func = True + + if starting_func: + body_found = False + for start_linenum in xrange(linenum, clean_lines.NumLines()): + start_line = lines[start_linenum] + joined_line += ' ' + start_line.lstrip() + if Search(r'(;|})', start_line): # Declarations and trivial functions + body_found = True + break # ... ignore + elif Search(r'{', start_line): + body_found = True + function = Search(r'((\w|:)*)\(', line).group(1) + if Match(r'TEST', function): # Handle TEST... macros + parameter_regexp = Search(r'(\(.*\))', joined_line) + if parameter_regexp: # Ignore bad syntax + function += parameter_regexp.group(1) + else: + function += '()' + function_state.Begin(function) + break + if not body_found: + # No body for the function (or evidence of a non-function) was found. + error(filename, linenum, 'readability/fn_size', 5, + 'Lint failed to find start of function body.') + elif Match(r'^\}\s*$', line): # function end + function_state.Check(error, filename, linenum) + function_state.End() + elif not Match(r'^\s*$', line): + function_state.Count() # Count non-blank/non-comment lines. + + +_RE_PATTERN_TODO = re.compile(r'^//(\s*)TODO(\(.+?\))?:?(\s|$)?') + + +def CheckComment(line, filename, linenum, next_line_start, error): + """Checks for common mistakes in comments. + + Args: + line: The line in question. + filename: The name of the current file. + linenum: The number of the line to check. + next_line_start: The first non-whitespace column of the next line. + error: The function to call with any errors found. + """ + commentpos = line.find('//') + if commentpos != -1: + # Check if the // may be in quotes. If so, ignore it + if re.sub(r'\\.', '', line[0:commentpos]).count('"') % 2 == 0: + # Allow one space for new scopes, two spaces otherwise: + if (not (Match(r'^.*{ *//', line) and next_line_start == commentpos) and + ((commentpos >= 1 and + line[commentpos-1] not in string.whitespace) or + (commentpos >= 2 and + line[commentpos-2] not in string.whitespace))): + error(filename, linenum, 'whitespace/comments', 2, + 'At least two spaces is best between code and comments') + + # Checks for common mistakes in TODO comments. + comment = line[commentpos:] + match = _RE_PATTERN_TODO.match(comment) + if match: + # One whitespace is correct; zero whitespace is handled elsewhere. + leading_whitespace = match.group(1) + if len(leading_whitespace) > 1: + error(filename, linenum, 'whitespace/todo', 2, + 'Too many spaces before TODO') + + username = match.group(2) + if not username: + error(filename, linenum, 'readability/todo', 2, + 'Missing username in TODO; it should look like ' + '"// TODO(my_username): Stuff."') + + middle_whitespace = match.group(3) + # Comparisons made explicit for correctness -- pylint: disable=g-explicit-bool-comparison + if middle_whitespace != ' ' and middle_whitespace != '': + error(filename, linenum, 'whitespace/todo', 2, + 'TODO(my_username) should be followed by a space') + + # If the comment contains an alphanumeric character, there + # should be a space somewhere between it and the // unless + # it's a /// or //! Doxygen comment. + if (Match(r'//[^ ]*\w', comment) and + not Match(r'(///|//\!)(\s+|$)', comment)): + error(filename, linenum, 'whitespace/comments', 4, + 'Should have a space between // and comment') + + +def CheckSpacing(filename, clean_lines, linenum, nesting_state, error): + """Checks for the correctness of various spacing issues in the code. + + Things we check for: spaces around operators, spaces after + if/for/while/switch, no spaces around parens in function calls, two + spaces between code and comment, don't start a block with a blank + line, don't end a function with a blank line, don't add a blank line + after public/protected/private, don't have too many blank lines in a row. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: The function to call with any errors found. + """ + + # Don't use "elided" lines here, otherwise we can't check commented lines. + # Don't want to use "raw" either, because we don't want to check inside C++11 + # raw strings, + raw = clean_lines.lines_without_raw_strings + line = raw[linenum] + + # Before nixing comments, check if the line is blank for no good + # reason. This includes the first line after a block is opened, and + # blank lines at the end of a function (ie, right before a line like '}' + # + # Skip all the blank line checks if we are immediately inside a + # namespace body. In other words, don't issue blank line warnings + # for this block: + # namespace { + # + # } + # + # A warning about missing end of namespace comments will be issued instead. + # + # Also skip blank line checks for 'extern "C"' blocks, which are formatted + # like namespaces. + if (IsBlankLine(line) and + not nesting_state.InNamespaceBody() and + not nesting_state.InExternC()): + elided = clean_lines.elided + prev_line = elided[linenum - 1] + prevbrace = prev_line.rfind('{') + # TODO(unknown): Don't complain if line before blank line, and line after, + # both start with alnums and are indented the same amount. + # This ignores whitespace at the start of a namespace block + # because those are not usually indented. + if prevbrace != -1 and prev_line[prevbrace:].find('}') == -1: + # OK, we have a blank line at the start of a code block. Before we + # complain, we check if it is an exception to the rule: The previous + # non-empty line has the parameters of a function header that are indented + # 4 spaces (because they did not fit in a 80 column line when placed on + # the same line as the function name). We also check for the case where + # the previous line is indented 6 spaces, which may happen when the + # initializers of a constructor do not fit into a 80 column line. + exception = False + if Match(r' {6}\w', prev_line): # Initializer list? + # We are looking for the opening column of initializer list, which + # should be indented 4 spaces to cause 6 space indentation afterwards. + search_position = linenum-2 + while (search_position >= 0 + and Match(r' {6}\w', elided[search_position])): + search_position -= 1 + exception = (search_position >= 0 + and elided[search_position][:5] == ' :') + else: + # Search for the function arguments or an initializer list. We use a + # simple heuristic here: If the line is indented 4 spaces; and we have a + # closing paren, without the opening paren, followed by an opening brace + # or colon (for initializer lists) we assume that it is the last line of + # a function header. If we have a colon indented 4 spaces, it is an + # initializer list. + exception = (Match(r' {4}\w[^\(]*\)\s*(const\s*)?(\{\s*$|:)', + prev_line) + or Match(r' {4}:', prev_line)) + + if not exception: + error(filename, linenum, 'whitespace/blank_line', 2, + 'Redundant blank line at the start of a code block ' + 'should be deleted.') + # Ignore blank lines at the end of a block in a long if-else + # chain, like this: + # if (condition1) { + # // Something followed by a blank line + # + # } else if (condition2) { + # // Something else + # } + if linenum + 1 < clean_lines.NumLines(): + next_line = raw[linenum + 1] + if (next_line + and Match(r'\s*}', next_line) + and next_line.find('} else ') == -1): + error(filename, linenum, 'whitespace/blank_line', 3, + 'Redundant blank line at the end of a code block ' + 'should be deleted.') + + matched = Match(r'\s*(public|protected|private):', prev_line) + if matched: + error(filename, linenum, 'whitespace/blank_line', 3, + 'Do not leave a blank line after "%s:"' % matched.group(1)) + + # Next, check comments + next_line_start = 0 + if linenum + 1 < clean_lines.NumLines(): + next_line = raw[linenum + 1] + next_line_start = len(next_line) - len(next_line.lstrip()) + CheckComment(line, filename, linenum, next_line_start, error) + + # get rid of comments and strings + line = clean_lines.elided[linenum] + + # You shouldn't have spaces before your brackets, except maybe after + # 'delete []' or 'return []() {};' + if Search(r'\w\s+\[', line) and not Search(r'(?:delete|return)\s+\[', line): + error(filename, linenum, 'whitespace/braces', 5, + 'Extra space before [') + + # In range-based for, we wanted spaces before and after the colon, but + # not around "::" tokens that might appear. + if (Search(r'for *\(.*[^:]:[^: ]', line) or + Search(r'for *\(.*[^: ]:[^:]', line)): + error(filename, linenum, 'whitespace/forcolon', 2, + 'Missing space around colon in range-based for loop') + + +def CheckOperatorSpacing(filename, clean_lines, linenum, error): + """Checks for horizontal spacing around operators. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Don't try to do spacing checks for operator methods. Do this by + # replacing the troublesome characters with something else, + # preserving column position for all other characters. + # + # The replacement is done repeatedly to avoid false positives from + # operators that call operators. + while True: + match = Match(r'^(.*\boperator\b)(\S+)(\s*\(.*)$', line) + if match: + line = match.group(1) + ('_' * len(match.group(2))) + match.group(3) + else: + break + + # We allow no-spaces around = within an if: "if ( (a=Foo()) == 0 )". + # Otherwise not. Note we only check for non-spaces on *both* sides; + # sometimes people put non-spaces on one side when aligning ='s among + # many lines (not that this is behavior that I approve of...) + if ((Search(r'[\w.]=', line) or + Search(r'=[\w.]', line)) + and not Search(r'\b(if|while|for) ', line) + # Operators taken from [lex.operators] in C++11 standard. + and not Search(r'(>=|<=|==|!=|&=|\^=|\|=|\+=|\*=|\/=|\%=)', line) + and not Search(r'operator=', line)): + error(filename, linenum, 'whitespace/operators', 4, + 'Missing spaces around =') + + # It's ok not to have spaces around binary operators like + - * /, but if + # there's too little whitespace, we get concerned. It's hard to tell, + # though, so we punt on this one for now. TODO. + + # You should always have whitespace around binary operators. + # + # Check <= and >= first to avoid false positives with < and >, then + # check non-include lines for spacing around < and >. + # + # If the operator is followed by a comma, assume it's be used in a + # macro context and don't do any checks. This avoids false + # positives. + # + # Note that && is not included here. This is because there are too + # many false positives due to RValue references. + match = Search(r'[^<>=!\s](==|!=|<=|>=|\|\|)[^<>=!\s,;\)]', line) + if match: + error(filename, linenum, 'whitespace/operators', 3, + 'Missing spaces around %s' % match.group(1)) + elif not Match(r'#.*include', line): + # Look for < that is not surrounded by spaces. This is only + # triggered if both sides are missing spaces, even though + # technically should should flag if at least one side is missing a + # space. This is done to avoid some false positives with shifts. + match = Match(r'^(.*[^\s<])<[^\s=<,]', line) + if match: + (_, _, end_pos) = CloseExpression( + clean_lines, linenum, len(match.group(1))) + if end_pos <= -1: + error(filename, linenum, 'whitespace/operators', 3, + 'Missing spaces around <') + + # Look for > that is not surrounded by spaces. Similar to the + # above, we only trigger if both sides are missing spaces to avoid + # false positives with shifts. + match = Match(r'^(.*[^-\s>])>[^\s=>,]', line) + if match: + (_, _, start_pos) = ReverseCloseExpression( + clean_lines, linenum, len(match.group(1))) + if start_pos <= -1: + error(filename, linenum, 'whitespace/operators', 3, + 'Missing spaces around >') + + # We allow no-spaces around << when used like this: 10<<20, but + # not otherwise (particularly, not when used as streams) + # + # We also allow operators following an opening parenthesis, since + # those tend to be macros that deal with operators. + match = Search(r'(operator|[^\s(<])(?:L|UL|LL|ULL|l|ul|ll|ull)?<<([^\s,=<])', line) + if (match and not (match.group(1).isdigit() and match.group(2).isdigit()) and + not (match.group(1) == 'operator' and match.group(2) == ';')): + error(filename, linenum, 'whitespace/operators', 3, + 'Missing spaces around <<') + + # We allow no-spaces around >> for almost anything. This is because + # C++11 allows ">>" to close nested templates, which accounts for + # most cases when ">>" is not followed by a space. + # + # We still warn on ">>" followed by alpha character, because that is + # likely due to ">>" being used for right shifts, e.g.: + # value >> alpha + # + # When ">>" is used to close templates, the alphanumeric letter that + # follows would be part of an identifier, and there should still be + # a space separating the template type and the identifier. + # type> alpha + match = Search(r'>>[a-zA-Z_]', line) + if match: + error(filename, linenum, 'whitespace/operators', 3, + 'Missing spaces around >>') + + # There shouldn't be space around unary operators + match = Search(r'(!\s|~\s|[\s]--[\s;]|[\s]\+\+[\s;])', line) + if match: + error(filename, linenum, 'whitespace/operators', 4, + 'Extra space for operator %s' % match.group(1)) + + +def CheckParenthesisSpacing(filename, clean_lines, linenum, error): + """Checks for horizontal spacing around parentheses. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # No spaces after an if, while, switch, or for + match = Search(r' (if\(|for\(|while\(|switch\()', line) + if match: + error(filename, linenum, 'whitespace/parens', 5, + 'Missing space before ( in %s' % match.group(1)) + + # For if/for/while/switch, the left and right parens should be + # consistent about how many spaces are inside the parens, and + # there should either be zero or one spaces inside the parens. + # We don't want: "if ( foo)" or "if ( foo )". + # Exception: "for ( ; foo; bar)" and "for (foo; bar; )" are allowed. + match = Search(r'\b(if|for|while|switch)\s*' + r'\(([ ]*)(.).*[^ ]+([ ]*)\)\s*{\s*$', + line) + if match: + if len(match.group(2)) != len(match.group(4)): + if not (match.group(3) == ';' and + len(match.group(2)) == 1 + len(match.group(4)) or + not match.group(2) and Search(r'\bfor\s*\(.*; \)', line)): + error(filename, linenum, 'whitespace/parens', 5, + 'Mismatching spaces inside () in %s' % match.group(1)) + if len(match.group(2)) not in [0, 1]: + error(filename, linenum, 'whitespace/parens', 5, + 'Should have zero or one spaces inside ( and ) in %s' % + match.group(1)) + + +def CheckCommaSpacing(filename, clean_lines, linenum, error): + """Checks for horizontal spacing near commas and semicolons. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + raw = clean_lines.lines_without_raw_strings + line = clean_lines.elided[linenum] + + # You should always have a space after a comma (either as fn arg or operator) + # + # This does not apply when the non-space character following the + # comma is another comma, since the only time when that happens is + # for empty macro arguments. + # + # We run this check in two passes: first pass on elided lines to + # verify that lines contain missing whitespaces, second pass on raw + # lines to confirm that those missing whitespaces are not due to + # elided comments. + if (Search(r',[^,\s]', ReplaceAll(r'\boperator\s*,\s*\(', 'F(', line)) and + Search(r',[^,\s]', raw[linenum])): + error(filename, linenum, 'whitespace/comma', 3, + 'Missing space after ,') + + # You should always have a space after a semicolon + # except for few corner cases + # TODO(unknown): clarify if 'if (1) { return 1;}' is requires one more + # space after ; + if Search(r';[^\s};\\)/]', line): + error(filename, linenum, 'whitespace/semicolon', 3, + 'Missing space after ;') + + +def _IsType(clean_lines, nesting_state, expr): + """Check if expression looks like a type name, returns true if so. + + Args: + clean_lines: A CleansedLines instance containing the file. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + expr: The expression to check. + Returns: + True, if token looks like a type. + """ + # Keep only the last token in the expression + last_word = Match(r'^.*(\b\S+)$', expr) + if last_word: + token = last_word.group(1) + else: + token = expr + + # Match native types and stdint types + if _TYPES.match(token): + return True + + # Try a bit harder to match templated types. Walk up the nesting + # stack until we find something that resembles a typename + # declaration for what we are looking for. + typename_pattern = (r'\b(?:typename|class|struct)\s+' + re.escape(token) + + r'\b') + block_index = len(nesting_state.stack) - 1 + while block_index >= 0: + if isinstance(nesting_state.stack[block_index], _NamespaceInfo): + return False + + # Found where the opening brace is. We want to scan from this + # line up to the beginning of the function, minus a few lines. + # template + # class C + # : public ... { // start scanning here + last_line = nesting_state.stack[block_index].starting_linenum + + next_block_start = 0 + if block_index > 0: + next_block_start = nesting_state.stack[block_index - 1].starting_linenum + first_line = last_line + while first_line >= next_block_start: + if clean_lines.elided[first_line].find('template') >= 0: + break + first_line -= 1 + if first_line < next_block_start: + # Didn't find any "template" keyword before reaching the next block, + # there are probably no template things to check for this block + block_index -= 1 + continue + + # Look for typename in the specified range + for i in xrange(first_line, last_line + 1, 1): + if Search(typename_pattern, clean_lines.elided[i]): + return True + block_index -= 1 + + return False + + +def CheckBracesSpacing(filename, clean_lines, linenum, nesting_state, error): + """Checks for horizontal spacing near commas. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Except after an opening paren, or after another opening brace (in case of + # an initializer list, for instance), you should have spaces before your + # braces when they are delimiting blocks, classes, namespaces etc. + # And since you should never have braces at the beginning of a line, + # this is an easy test. Except that braces used for initialization don't + # follow the same rule; we often don't want spaces before those. + match = Match(r'^(.*[^ ({>]){', line) + + if match: + # Try a bit harder to check for brace initialization. This + # happens in one of the following forms: + # Constructor() : initializer_list_{} { ... } + # Constructor{}.MemberFunction() + # Type variable{}; + # FunctionCall(type{}, ...); + # LastArgument(..., type{}); + # LOG(INFO) << type{} << " ..."; + # map_of_type[{...}] = ...; + # ternary = expr ? new type{} : nullptr; + # OuterTemplate{}> + # + # We check for the character following the closing brace, and + # silence the warning if it's one of those listed above, i.e. + # "{.;,)<>]:". + # + # To account for nested initializer list, we allow any number of + # closing braces up to "{;,)<". We can't simply silence the + # warning on first sight of closing brace, because that would + # cause false negatives for things that are not initializer lists. + # Silence this: But not this: + # Outer{ if (...) { + # Inner{...} if (...){ // Missing space before { + # }; } + # + # There is a false negative with this approach if people inserted + # spurious semicolons, e.g. "if (cond){};", but we will catch the + # spurious semicolon with a separate check. + leading_text = match.group(1) + (endline, endlinenum, endpos) = CloseExpression( + clean_lines, linenum, len(match.group(1))) + trailing_text = '' + if endpos > -1: + trailing_text = endline[endpos:] + for offset in xrange(endlinenum + 1, + min(endlinenum + 3, clean_lines.NumLines() - 1)): + trailing_text += clean_lines.elided[offset] + # We also suppress warnings for `uint64_t{expression}` etc., as the style + # guide recommends brace initialization for integral types to avoid + # overflow/truncation. + if (not Match(r'^[\s}]*[{.;,)<>\]:]', trailing_text) + and not _IsType(clean_lines, nesting_state, leading_text)): + error(filename, linenum, 'whitespace/braces', 5, + 'Missing space before {') + + # Make sure '} else {' has spaces. + if Search(r'}else', line): + error(filename, linenum, 'whitespace/braces', 5, + 'Missing space before else') + + # You shouldn't have a space before a semicolon at the end of the line. + # There's a special case for "for" since the style guide allows space before + # the semicolon there. + if Search(r':\s*;\s*$', line): + error(filename, linenum, 'whitespace/semicolon', 5, + 'Semicolon defining empty statement. Use {} instead.') + elif Search(r'^\s*;\s*$', line): + error(filename, linenum, 'whitespace/semicolon', 5, + 'Line contains only semicolon. If this should be an empty statement, ' + 'use {} instead.') + elif (Search(r'\s+;\s*$', line) and + not Search(r'\bfor\b', line)): + error(filename, linenum, 'whitespace/semicolon', 5, + 'Extra space before last semicolon. If this should be an empty ' + 'statement, use {} instead.') + + +def IsDecltype(clean_lines, linenum, column): + """Check if the token ending on (linenum, column) is decltype(). + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: the number of the line to check. + column: end column of the token to check. + Returns: + True if this token is decltype() expression, False otherwise. + """ + (text, _, start_col) = ReverseCloseExpression(clean_lines, linenum, column) + if start_col < 0: + return False + if Search(r'\bdecltype\s*$', text[0:start_col]): + return True + return False + +def CheckSectionSpacing(filename, clean_lines, class_info, linenum, error): + """Checks for additional blank line issues related to sections. + + Currently the only thing checked here is blank line before protected/private. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + class_info: A _ClassInfo objects. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + # Skip checks if the class is small, where small means 25 lines or less. + # 25 lines seems like a good cutoff since that's the usual height of + # terminals, and any class that can't fit in one screen can't really + # be considered "small". + # + # Also skip checks if we are on the first line. This accounts for + # classes that look like + # class Foo { public: ... }; + # + # If we didn't find the end of the class, last_line would be zero, + # and the check will be skipped by the first condition. + if (class_info.last_line - class_info.starting_linenum <= 24 or + linenum <= class_info.starting_linenum): + return + + matched = Match(r'\s*(public|protected|private):', clean_lines.lines[linenum]) + if matched: + # Issue warning if the line before public/protected/private was + # not a blank line, but don't do this if the previous line contains + # "class" or "struct". This can happen two ways: + # - We are at the beginning of the class. + # - We are forward-declaring an inner class that is semantically + # private, but needed to be public for implementation reasons. + # Also ignores cases where the previous line ends with a backslash as can be + # common when defining classes in C macros. + prev_line = clean_lines.lines[linenum - 1] + if (not IsBlankLine(prev_line) and + not Search(r'\b(class|struct)\b', prev_line) and + not Search(r'\\$', prev_line)): + # Try a bit harder to find the beginning of the class. This is to + # account for multi-line base-specifier lists, e.g.: + # class Derived + # : public Base { + end_class_head = class_info.starting_linenum + for i in range(class_info.starting_linenum, linenum): + if Search(r'\{\s*$', clean_lines.lines[i]): + end_class_head = i + break + if end_class_head < linenum - 1: + error(filename, linenum, 'whitespace/blank_line', 3, + '"%s:" should be preceded by a blank line' % matched.group(1)) + + +def GetPreviousNonBlankLine(clean_lines, linenum): + """Return the most recent non-blank line and its line number. + + Args: + clean_lines: A CleansedLines instance containing the file contents. + linenum: The number of the line to check. + + Returns: + A tuple with two elements. The first element is the contents of the last + non-blank line before the current line, or the empty string if this is the + first non-blank line. The second is the line number of that line, or -1 + if this is the first non-blank line. + """ + + prevlinenum = linenum - 1 + while prevlinenum >= 0: + prevline = clean_lines.elided[prevlinenum] + if not IsBlankLine(prevline): # if not a blank line... + return (prevline, prevlinenum) + prevlinenum -= 1 + return ('', -1) + + +def CheckBraces(filename, clean_lines, linenum, error): + """Looks for misplaced braces (e.g. at the end of line). + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + + line = clean_lines.elided[linenum] # get rid of comments and strings + + if Match(r'\s*{\s*$', line): + # We allow an open brace to start a line in the case where someone is using + # braces in a block to explicitly create a new scope, which is commonly used + # to control the lifetime of stack-allocated variables. Braces are also + # used for brace initializers inside function calls. We don't detect this + # perfectly: we just don't complain if the last non-whitespace character on + # the previous non-blank line is ',', ';', ':', '(', '{', or '}', or if the + # previous line starts a preprocessor block. We also allow a brace on the + # following line if it is part of an array initialization and would not fit + # within the 80 character limit of the preceding line. + prevline = GetPreviousNonBlankLine(clean_lines, linenum)[0] + if (not Search(r'[,;:}{(]\s*$', prevline) and + not Match(r'\s*#', prevline) and + not (GetLineWidth(prevline) > _line_length - 2 and '[]' in prevline)): + error(filename, linenum, 'whitespace/braces', 4, + '{ should almost always be at the end of the previous line') + + # An else clause should be on the same line as the preceding closing brace. + if Match(r'\s*else\b\s*(?:if\b|\{|$)', line): + prevline = GetPreviousNonBlankLine(clean_lines, linenum)[0] + if Match(r'\s*}\s*$', prevline): + error(filename, linenum, 'whitespace/newline', 4, + 'An else should appear on the same line as the preceding }') + + # If braces come on one side of an else, they should be on both. + # However, we have to worry about "else if" that spans multiple lines! + if Search(r'else if\s*\(', line): # could be multi-line if + brace_on_left = bool(Search(r'}\s*else if\s*\(', line)) + # find the ( after the if + pos = line.find('else if') + pos = line.find('(', pos) + if pos > 0: + (endline, _, endpos) = CloseExpression(clean_lines, linenum, pos) + brace_on_right = endline[endpos:].find('{') != -1 + if brace_on_left != brace_on_right: # must be brace after if + error(filename, linenum, 'readability/braces', 5, + 'If an else has a brace on one side, it should have it on both') + elif Search(r'}\s*else[^{]*$', line) or Match(r'[^}]*else\s*{', line): + error(filename, linenum, 'readability/braces', 5, + 'If an else has a brace on one side, it should have it on both') + + # Likewise, an else should never have the else clause on the same line + if Search(r'\belse [^\s{]', line) and not Search(r'\belse if\b', line): + error(filename, linenum, 'whitespace/newline', 4, + 'Else clause should never be on same line as else (use 2 lines)') + + # In the same way, a do/while should never be on one line + if Match(r'\s*do [^\s{]', line): + error(filename, linenum, 'whitespace/newline', 4, + 'do/while clauses should not be on a single line') + + # Check single-line if/else bodies. The style guide says 'curly braces are not + # required for single-line statements'. We additionally allow multi-line, + # single statements, but we reject anything with more than one semicolon in + # it. This means that the first semicolon after the if should be at the end of + # its line, and the line after that should have an indent level equal to or + # lower than the if. We also check for ambiguous if/else nesting without + # braces. + if_else_match = Search(r'\b(if\s*\(|else\b)', line) + if if_else_match and not Match(r'\s*#', line): + if_indent = GetIndentLevel(line) + endline, endlinenum, endpos = line, linenum, if_else_match.end() + if_match = Search(r'\bif\s*\(', line) + if if_match: + # This could be a multiline if condition, so find the end first. + pos = if_match.end() - 1 + (endline, endlinenum, endpos) = CloseExpression(clean_lines, linenum, pos) + # Check for an opening brace, either directly after the if or on the next + # line. If found, this isn't a single-statement conditional. + if (not Match(r'\s*{', endline[endpos:]) + and not (Match(r'\s*$', endline[endpos:]) + and endlinenum < (len(clean_lines.elided) - 1) + and Match(r'\s*{', clean_lines.elided[endlinenum + 1]))): + while (endlinenum < len(clean_lines.elided) + and ';' not in clean_lines.elided[endlinenum][endpos:]): + endlinenum += 1 + endpos = 0 + if endlinenum < len(clean_lines.elided): + endline = clean_lines.elided[endlinenum] + # We allow a mix of whitespace and closing braces (e.g. for one-liner + # methods) and a single \ after the semicolon (for macros) + endpos = endline.find(';') + if not Match(r';[\s}]*(\\?)$', endline[endpos:]): + # Semicolon isn't the last character, there's something trailing. + # Output a warning if the semicolon is not contained inside + # a lambda expression. + if not Match(r'^[^{};]*\[[^\[\]]*\][^{}]*\{[^{}]*\}\s*\)*[;,]\s*$', + endline): + error(filename, linenum, 'readability/braces', 4, + 'If/else bodies with multiple statements require braces') + elif endlinenum < len(clean_lines.elided) - 1: + # Make sure the next line is dedented + next_line = clean_lines.elided[endlinenum + 1] + next_indent = GetIndentLevel(next_line) + # With ambiguous nested if statements, this will error out on the + # if that *doesn't* match the else, regardless of whether it's the + # inner one or outer one. + if (if_match and Match(r'\s*else\b', next_line) + and next_indent != if_indent): + error(filename, linenum, 'readability/braces', 4, + 'Else clause should be indented at the same level as if. ' + 'Ambiguous nested if/else chains require braces.') + elif next_indent > if_indent: + error(filename, linenum, 'readability/braces', 4, + 'If/else bodies with multiple statements require braces') + + +def CheckTrailingSemicolon(filename, clean_lines, linenum, error): + """Looks for redundant trailing semicolon. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + + line = clean_lines.elided[linenum] + + # Block bodies should not be followed by a semicolon. Due to C++11 + # brace initialization, there are more places where semicolons are + # required than not, so we use a whitelist approach to check these + # rather than a blacklist. These are the places where "};" should + # be replaced by just "}": + # 1. Some flavor of block following closing parenthesis: + # for (;;) {}; + # while (...) {}; + # switch (...) {}; + # Function(...) {}; + # if (...) {}; + # if (...) else if (...) {}; + # + # 2. else block: + # if (...) else {}; + # + # 3. const member function: + # Function(...) const {}; + # + # 4. Block following some statement: + # x = 42; + # {}; + # + # 5. Block at the beginning of a function: + # Function(...) { + # {}; + # } + # + # Note that naively checking for the preceding "{" will also match + # braces inside multi-dimensional arrays, but this is fine since + # that expression will not contain semicolons. + # + # 6. Block following another block: + # while (true) {} + # {}; + # + # 7. End of namespaces: + # namespace {}; + # + # These semicolons seems far more common than other kinds of + # redundant semicolons, possibly due to people converting classes + # to namespaces. For now we do not warn for this case. + # + # Try matching case 1 first. + match = Match(r'^(.*\)\s*)\{', line) + if match: + # Matched closing parenthesis (case 1). Check the token before the + # matching opening parenthesis, and don't warn if it looks like a + # macro. This avoids these false positives: + # - macro that defines a base class + # - multi-line macro that defines a base class + # - macro that defines the whole class-head + # + # But we still issue warnings for macros that we know are safe to + # warn, specifically: + # - TEST, TEST_F, TEST_P, MATCHER, MATCHER_P + # - TYPED_TEST + # - INTERFACE_DEF + # - EXCLUSIVE_LOCKS_REQUIRED, SHARED_LOCKS_REQUIRED, LOCKS_EXCLUDED: + # + # We implement a whitelist of safe macros instead of a blacklist of + # unsafe macros, even though the latter appears less frequently in + # google code and would have been easier to implement. This is because + # the downside for getting the whitelist wrong means some extra + # semicolons, while the downside for getting the blacklist wrong + # would result in compile errors. + # + # In addition to macros, we also don't want to warn on + # - Compound literals + # - Lambdas + # - alignas specifier with anonymous structs + # - decltype + closing_brace_pos = match.group(1).rfind(')') + opening_parenthesis = ReverseCloseExpression( + clean_lines, linenum, closing_brace_pos) + if opening_parenthesis[2] > -1: + line_prefix = opening_parenthesis[0][0:opening_parenthesis[2]] + macro = Search(r'\b([A-Z_][A-Z0-9_]*)\s*$', line_prefix) + func = Match(r'^(.*\])\s*$', line_prefix) + if ((macro and + macro.group(1) not in ( + 'TEST', 'TEST_F', 'MATCHER', 'MATCHER_P', 'TYPED_TEST', + 'EXCLUSIVE_LOCKS_REQUIRED', 'SHARED_LOCKS_REQUIRED', + 'LOCKS_EXCLUDED', 'INTERFACE_DEF')) or + (func and not Search(r'\boperator\s*\[\s*\]', func.group(1))) or + Search(r'\b(?:struct|union)\s+alignas\s*$', line_prefix) or + Search(r'\bdecltype$', line_prefix) or + Search(r'\s+=\s*$', line_prefix)): + match = None + if (match and + opening_parenthesis[1] > 1 and + Search(r'\]\s*$', clean_lines.elided[opening_parenthesis[1] - 1])): + # Multi-line lambda-expression + match = None + + else: + # Try matching cases 2-3. + match = Match(r'^(.*(?:else|\)\s*const)\s*)\{', line) + if not match: + # Try matching cases 4-6. These are always matched on separate lines. + # + # Note that we can't simply concatenate the previous line to the + # current line and do a single match, otherwise we may output + # duplicate warnings for the blank line case: + # if (cond) { + # // blank line + # } + prevline = GetPreviousNonBlankLine(clean_lines, linenum)[0] + if prevline and Search(r'[;{}]\s*$', prevline): + match = Match(r'^(\s*)\{', line) + + # Check matching closing brace + if match: + (endline, endlinenum, endpos) = CloseExpression( + clean_lines, linenum, len(match.group(1))) + if endpos > -1 and Match(r'^\s*;', endline[endpos:]): + # Current {} pair is eligible for semicolon check, and we have found + # the redundant semicolon, output warning here. + # + # Note: because we are scanning forward for opening braces, and + # outputting warnings for the matching closing brace, if there are + # nested blocks with trailing semicolons, we will get the error + # messages in reversed order. + + # We need to check the line forward for NOLINT + raw_lines = clean_lines.raw_lines + ParseNolintSuppressions(filename, raw_lines[endlinenum-1], endlinenum-1, + error) + ParseNolintSuppressions(filename, raw_lines[endlinenum], endlinenum, + error) + + error(filename, endlinenum, 'readability/braces', 4, + "You don't need a ; after a }") + + +def CheckEmptyBlockBody(filename, clean_lines, linenum, error): + """Look for empty loop/conditional body with only a single semicolon. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + + # Search for loop keywords at the beginning of the line. Because only + # whitespaces are allowed before the keywords, this will also ignore most + # do-while-loops, since those lines should start with closing brace. + # + # We also check "if" blocks here, since an empty conditional block + # is likely an error. + line = clean_lines.elided[linenum] + matched = Match(r'\s*(for|while|if)\s*\(', line) + if matched: + # Find the end of the conditional expression. + (end_line, end_linenum, end_pos) = CloseExpression( + clean_lines, linenum, line.find('(')) + + # Output warning if what follows the condition expression is a semicolon. + # No warning for all other cases, including whitespace or newline, since we + # have a separate check for semicolons preceded by whitespace. + if end_pos >= 0 and Match(r';', end_line[end_pos:]): + if matched.group(1) == 'if': + error(filename, end_linenum, 'whitespace/empty_conditional_body', 5, + 'Empty conditional bodies should use {}') + else: + error(filename, end_linenum, 'whitespace/empty_loop_body', 5, + 'Empty loop bodies should use {} or continue') + + # Check for if statements that have completely empty bodies (no comments) + # and no else clauses. + if end_pos >= 0 and matched.group(1) == 'if': + # Find the position of the opening { for the if statement. + # Return without logging an error if it has no brackets. + opening_linenum = end_linenum + opening_line_fragment = end_line[end_pos:] + # Loop until EOF or find anything that's not whitespace or opening {. + while not Search(r'^\s*\{', opening_line_fragment): + if Search(r'^(?!\s*$)', opening_line_fragment): + # Conditional has no brackets. + return + opening_linenum += 1 + if opening_linenum == len(clean_lines.elided): + # Couldn't find conditional's opening { or any code before EOF. + return + opening_line_fragment = clean_lines.elided[opening_linenum] + # Set opening_line (opening_line_fragment may not be entire opening line). + opening_line = clean_lines.elided[opening_linenum] + + # Find the position of the closing }. + opening_pos = opening_line_fragment.find('{') + if opening_linenum == end_linenum: + # We need to make opening_pos relative to the start of the entire line. + opening_pos += end_pos + (closing_line, closing_linenum, closing_pos) = CloseExpression( + clean_lines, opening_linenum, opening_pos) + if closing_pos < 0: + return + + # Now construct the body of the conditional. This consists of the portion + # of the opening line after the {, all lines until the closing line, + # and the portion of the closing line before the }. + if (clean_lines.raw_lines[opening_linenum] != + CleanseComments(clean_lines.raw_lines[opening_linenum])): + # Opening line ends with a comment, so conditional isn't empty. + return + if closing_linenum > opening_linenum: + # Opening line after the {. Ignore comments here since we checked above. + bodylist = list(opening_line[opening_pos+1:]) + # All lines until closing line, excluding closing line, with comments. + bodylist.extend(clean_lines.raw_lines[opening_linenum+1:closing_linenum]) + # Closing line before the }. Won't (and can't) have comments. + bodylist.append(clean_lines.elided[closing_linenum][:closing_pos-1]) + body = '\n'.join(bodylist) + else: + # If statement has brackets and fits on a single line. + body = opening_line[opening_pos+1:closing_pos-1] + + # Check if the body is empty + if not _EMPTY_CONDITIONAL_BODY_PATTERN.search(body): + return + # The body is empty. Now make sure there's not an else clause. + current_linenum = closing_linenum + current_line_fragment = closing_line[closing_pos:] + # Loop until EOF or find anything that's not whitespace or else clause. + while Search(r'^\s*$|^(?=\s*else)', current_line_fragment): + if Search(r'^(?=\s*else)', current_line_fragment): + # Found an else clause, so don't log an error. + return + current_linenum += 1 + if current_linenum == len(clean_lines.elided): + break + current_line_fragment = clean_lines.elided[current_linenum] + + # The body is empty and there's no else clause until EOF or other code. + error(filename, end_linenum, 'whitespace/empty_if_body', 4, + ('If statement had no body and no else clause')) + + +def FindCheckMacro(line): + """Find a replaceable CHECK-like macro. + + Args: + line: line to search on. + Returns: + (macro name, start position), or (None, -1) if no replaceable + macro is found. + """ + for macro in _CHECK_MACROS: + i = line.find(macro) + if i >= 0: + # Find opening parenthesis. Do a regular expression match here + # to make sure that we are matching the expected CHECK macro, as + # opposed to some other macro that happens to contain the CHECK + # substring. + matched = Match(r'^(.*\b' + macro + r'\s*)\(', line) + if not matched: + continue + return (macro, len(matched.group(1))) + return (None, -1) + + +def CheckCheck(filename, clean_lines, linenum, error): + """Checks the use of CHECK and EXPECT macros. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + + # Decide the set of replacement macros that should be suggested + lines = clean_lines.elided + (check_macro, start_pos) = FindCheckMacro(lines[linenum]) + if not check_macro: + return + + # Find end of the boolean expression by matching parentheses + (last_line, end_line, end_pos) = CloseExpression( + clean_lines, linenum, start_pos) + if end_pos < 0: + return + + # If the check macro is followed by something other than a + # semicolon, assume users will log their own custom error messages + # and don't suggest any replacements. + if not Match(r'\s*;', last_line[end_pos:]): + return + + if linenum == end_line: + expression = lines[linenum][start_pos + 1:end_pos - 1] + else: + expression = lines[linenum][start_pos + 1:] + for i in xrange(linenum + 1, end_line): + expression += lines[i] + expression += last_line[0:end_pos - 1] + + # Parse expression so that we can take parentheses into account. + # This avoids false positives for inputs like "CHECK((a < 4) == b)", + # which is not replaceable by CHECK_LE. + lhs = '' + rhs = '' + operator = None + while expression: + matched = Match(r'^\s*(<<|<<=|>>|>>=|->\*|->|&&|\|\||' + r'==|!=|>=|>|<=|<|\()(.*)$', expression) + if matched: + token = matched.group(1) + if token == '(': + # Parenthesized operand + expression = matched.group(2) + (end, _) = FindEndOfExpressionInLine(expression, 0, ['(']) + if end < 0: + return # Unmatched parenthesis + lhs += '(' + expression[0:end] + expression = expression[end:] + elif token in ('&&', '||'): + # Logical and/or operators. This means the expression + # contains more than one term, for example: + # CHECK(42 < a && a < b); + # + # These are not replaceable with CHECK_LE, so bail out early. + return + elif token in ('<<', '<<=', '>>', '>>=', '->*', '->'): + # Non-relational operator + lhs += token + expression = matched.group(2) + else: + # Relational operator + operator = token + rhs = matched.group(2) + break + else: + # Unparenthesized operand. Instead of appending to lhs one character + # at a time, we do another regular expression match to consume several + # characters at once if possible. Trivial benchmark shows that this + # is more efficient when the operands are longer than a single + # character, which is generally the case. + matched = Match(r'^([^-=!<>()&|]+)(.*)$', expression) + if not matched: + matched = Match(r'^(\s*\S)(.*)$', expression) + if not matched: + break + lhs += matched.group(1) + expression = matched.group(2) + + # Only apply checks if we got all parts of the boolean expression + if not (lhs and operator and rhs): + return + + # Check that rhs do not contain logical operators. We already know + # that lhs is fine since the loop above parses out && and ||. + if rhs.find('&&') > -1 or rhs.find('||') > -1: + return + + # At least one of the operands must be a constant literal. This is + # to avoid suggesting replacements for unprintable things like + # CHECK(variable != iterator) + # + # The following pattern matches decimal, hex integers, strings, and + # characters (in that order). + lhs = lhs.strip() + rhs = rhs.strip() + match_constant = r'^([-+]?(\d+|0[xX][0-9a-fA-F]+)[lLuU]{0,3}|".*"|\'.*\')$' + if Match(match_constant, lhs) or Match(match_constant, rhs): + # Note: since we know both lhs and rhs, we can provide a more + # descriptive error message like: + # Consider using CHECK_EQ(x, 42) instead of CHECK(x == 42) + # Instead of: + # Consider using CHECK_EQ instead of CHECK(a == b) + # + # We are still keeping the less descriptive message because if lhs + # or rhs gets long, the error message might become unreadable. + error(filename, linenum, 'readability/check', 2, + 'Consider using %s instead of %s(a %s b)' % ( + _CHECK_REPLACEMENT[check_macro][operator], + check_macro, operator)) + + +def CheckAltTokens(filename, clean_lines, linenum, error): + """Check alternative keywords being used in boolean expressions. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Avoid preprocessor lines + if Match(r'^\s*#', line): + return + + # Last ditch effort to avoid multi-line comments. This will not help + # if the comment started before the current line or ended after the + # current line, but it catches most of the false positives. At least, + # it provides a way to workaround this warning for people who use + # multi-line comments in preprocessor macros. + # + # TODO(unknown): remove this once cpplint has better support for + # multi-line comments. + if line.find('/*') >= 0 or line.find('*/') >= 0: + return + + for match in _ALT_TOKEN_REPLACEMENT_PATTERN.finditer(line): + error(filename, linenum, 'readability/alt_tokens', 2, + 'Use operator %s instead of %s' % ( + _ALT_TOKEN_REPLACEMENT[match.group(1)], match.group(1))) + + +def GetLineWidth(line): + """Determines the width of the line in column positions. + + Args: + line: A string, which may be a Unicode string. + + Returns: + The width of the line in column positions, accounting for Unicode + combining characters and wide characters. + """ + if isinstance(line, unicode): + width = 0 + for uc in unicodedata.normalize('NFC', line): + if unicodedata.east_asian_width(uc) in ('W', 'F'): + width += 2 + elif not unicodedata.combining(uc): + # Issue 337 + # https://mail.python.org/pipermail/python-list/2012-August/628809.html + if (sys.version_info.major, sys.version_info.minor) <= (3, 2): + # https://github.com/python/cpython/blob/2.7/Include/unicodeobject.h#L81 + is_wide_build = sysconfig.get_config_var("Py_UNICODE_SIZE") >= 4 + # https://github.com/python/cpython/blob/2.7/Objects/unicodeobject.c#L564 + is_low_surrogate = 0xDC00 <= ord(uc) <= 0xDFFF + if not is_wide_build and is_low_surrogate: + width -= 1 + + width += 1 + return width + else: + return len(line) + + +def CheckStyle(filename, clean_lines, linenum, file_extension, nesting_state, + error): + """Checks rules from the 'C++ style rules' section of cppguide.html. + + Most of these rules are hard to test (naming, comment style), but we + do what we can. In particular we check for 2-space indents, line lengths, + tab usage, spaces inside code, etc. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + file_extension: The extension (without the dot) of the filename. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: The function to call with any errors found. + """ + + # Don't use "elided" lines here, otherwise we can't check commented lines. + # Don't want to use "raw" either, because we don't want to check inside C++11 + # raw strings, + raw_lines = clean_lines.lines_without_raw_strings + line = raw_lines[linenum] + prev = raw_lines[linenum - 1] if linenum > 0 else '' + + if line.find('\t') != -1: + error(filename, linenum, 'whitespace/tab', 1, + 'Tab found; better to use spaces') + + # One or three blank spaces at the beginning of the line is weird; it's + # hard to reconcile that with 2-space indents. + # NOTE: here are the conditions rob pike used for his tests. Mine aren't + # as sophisticated, but it may be worth becoming so: RLENGTH==initial_spaces + # if(RLENGTH > 20) complain = 0; + # if(match($0, " +(error|private|public|protected):")) complain = 0; + # if(match(prev, "&& *$")) complain = 0; + # if(match(prev, "\\|\\| *$")) complain = 0; + # if(match(prev, "[\",=><] *$")) complain = 0; + # if(match($0, " <<")) complain = 0; + # if(match(prev, " +for \\(")) complain = 0; + # if(prevodd && match(prevprev, " +for \\(")) complain = 0; + scope_or_label_pattern = r'\s*\w+\s*:\s*\\?$' + classinfo = nesting_state.InnermostClass() + initial_spaces = 0 + cleansed_line = clean_lines.elided[linenum] + while initial_spaces < len(line) and line[initial_spaces] == ' ': + initial_spaces += 1 + # There are certain situations we allow one space, notably for + # section labels, and also lines containing multi-line raw strings. + # We also don't check for lines that look like continuation lines + # (of lines ending in double quotes, commas, equals, or angle brackets) + # because the rules for how to indent those are non-trivial. + if (not Search(r'[",=><] *$', prev) and + (initial_spaces == 1 or initial_spaces == 3) and + not Match(scope_or_label_pattern, cleansed_line) and + not (clean_lines.raw_lines[linenum] != line and + Match(r'^\s*""', line))): + error(filename, linenum, 'whitespace/indent', 3, + 'Weird number of spaces at line-start. ' + 'Are you using a 2-space indent?') + + if line and line[-1].isspace(): + error(filename, linenum, 'whitespace/end_of_line', 4, + 'Line ends in whitespace. Consider deleting these extra spaces.') + + # Check if the line is a header guard. + is_header_guard = False + if IsHeaderExtension(file_extension): + cppvar = GetHeaderGuardCPPVariable(filename) + if (line.startswith('#ifndef %s' % cppvar) or + line.startswith('#define %s' % cppvar) or + line.startswith('#endif // %s' % cppvar)): + is_header_guard = True + # #include lines and header guards can be long, since there's no clean way to + # split them. + # + # URLs can be long too. It's possible to split these, but it makes them + # harder to cut&paste. + # + # The "$Id:...$" comment may also get very long without it being the + # developers fault. + # + # Doxygen documentation copying can get pretty long when using an overloaded + # function declaration + if (not line.startswith('#include') and not is_header_guard and + not Match(r'^\s*//.*http(s?)://\S*$', line) and + not Match(r'^\s*//\s*[^\s]*$', line) and + not Match(r'^// \$Id:.*#[0-9]+ \$$', line) and + not Match(r'^\s*/// [@\\](copydoc|copydetails|copybrief) .*$', line)): + line_width = GetLineWidth(line) + if line_width > _line_length: + error(filename, linenum, 'whitespace/line_length', 2, + 'Lines should be <= %i characters long' % _line_length) + + if (cleansed_line.count(';') > 1 and + # allow simple single line lambdas + not Match(r'^[^{};]*\[[^\[\]]*\][^{}]*\{[^{}\n\r]*\}', + line) and + # for loops are allowed two ;'s (and may run over two lines). + cleansed_line.find('for') == -1 and + (GetPreviousNonBlankLine(clean_lines, linenum)[0].find('for') == -1 or + GetPreviousNonBlankLine(clean_lines, linenum)[0].find(';') != -1) and + # It's ok to have many commands in a switch case that fits in 1 line + not ((cleansed_line.find('case ') != -1 or + cleansed_line.find('default:') != -1) and + cleansed_line.find('break;') != -1)): + error(filename, linenum, 'whitespace/newline', 0, + 'More than one command on the same line') + + # Some more style checks + CheckBraces(filename, clean_lines, linenum, error) + CheckTrailingSemicolon(filename, clean_lines, linenum, error) + CheckEmptyBlockBody(filename, clean_lines, linenum, error) + CheckSpacing(filename, clean_lines, linenum, nesting_state, error) + CheckOperatorSpacing(filename, clean_lines, linenum, error) + CheckParenthesisSpacing(filename, clean_lines, linenum, error) + CheckCommaSpacing(filename, clean_lines, linenum, error) + CheckBracesSpacing(filename, clean_lines, linenum, nesting_state, error) + CheckSpacingForFunctionCall(filename, clean_lines, linenum, error) + CheckCheck(filename, clean_lines, linenum, error) + CheckAltTokens(filename, clean_lines, linenum, error) + classinfo = nesting_state.InnermostClass() + if classinfo: + CheckSectionSpacing(filename, clean_lines, classinfo, linenum, error) + + +_RE_PATTERN_INCLUDE = re.compile(r'^\s*#\s*include\s*([<"])([^>"]*)[>"].*$') +# Matches the first component of a filename delimited by -s and _s. That is: +# _RE_FIRST_COMPONENT.match('foo').group(0) == 'foo' +# _RE_FIRST_COMPONENT.match('foo.cc').group(0) == 'foo' +# _RE_FIRST_COMPONENT.match('foo-bar_baz.cc').group(0) == 'foo' +# _RE_FIRST_COMPONENT.match('foo_bar-baz.cc').group(0) == 'foo' +_RE_FIRST_COMPONENT = re.compile(r'^[^-_.]+') + + +def _DropCommonSuffixes(filename): + """Drops common suffixes like _test.cc or -inl.h from filename. + + For example: + >>> _DropCommonSuffixes('foo/foo-inl.h') + 'foo/foo' + >>> _DropCommonSuffixes('foo/bar/foo.cc') + 'foo/bar/foo' + >>> _DropCommonSuffixes('foo/foo_internal.h') + 'foo/foo' + >>> _DropCommonSuffixes('foo/foo_unusualinternal.h') + 'foo/foo_unusualinternal' + + Args: + filename: The input filename. + + Returns: + The filename with the common suffix removed. + """ + for suffix in itertools.chain( + ('%s.%s' % (test_suffix.lstrip('_'), ext) + for test_suffix, ext in itertools.product(_test_suffixes, GetNonHeaderExtensions())), + ('%s.%s' % (suffix, ext) + for suffix, ext in itertools.product(['inl', 'imp', 'internal'], GetHeaderExtensions()))): + if (filename.endswith(suffix) and len(filename) > len(suffix) and + filename[-len(suffix) - 1] in ('-', '_')): + return filename[:-len(suffix) - 1] + return os.path.splitext(filename)[0] + + +def _ClassifyInclude(fileinfo, include, is_system): + """Figures out what kind of header 'include' is. + + Args: + fileinfo: The current file cpplint is running over. A FileInfo instance. + include: The path to a #included file. + is_system: True if the #include used <> rather than "". + + Returns: + One of the _XXX_HEADER constants. + + For example: + >>> _ClassifyInclude(FileInfo('foo/foo.cc'), 'stdio.h', True) + _C_SYS_HEADER + >>> _ClassifyInclude(FileInfo('foo/foo.cc'), 'string', True) + _CPP_SYS_HEADER + >>> _ClassifyInclude(FileInfo('foo/foo.cc'), 'foo/foo.h', False) + _LIKELY_MY_HEADER + >>> _ClassifyInclude(FileInfo('foo/foo_unknown_extension.cc'), + ... 'bar/foo_other_ext.h', False) + _POSSIBLE_MY_HEADER + >>> _ClassifyInclude(FileInfo('foo/foo.cc'), 'foo/bar.h', False) + _OTHER_HEADER + """ + # This is a list of all standard c++ header files, except + # those already checked for above. + is_cpp_h = include in _CPP_HEADERS + + # Headers with C++ extensions shouldn't be considered C system headers + if is_system and os.path.splitext(include)[1] in ['.hpp', '.hxx', '.h++']: + is_system = False + + if is_system: + if is_cpp_h: + return _CPP_SYS_HEADER + else: + return _C_SYS_HEADER + + # If the target file and the include we're checking share a + # basename when we drop common extensions, and the include + # lives in . , then it's likely to be owned by the target file. + target_dir, target_base = ( + os.path.split(_DropCommonSuffixes(fileinfo.RepositoryName()))) + include_dir, include_base = os.path.split(_DropCommonSuffixes(include)) + target_dir_pub = os.path.normpath(target_dir + '/../public') + target_dir_pub = target_dir_pub.replace('\\', '/') + if target_base == include_base and ( + include_dir == target_dir or + include_dir == target_dir_pub): + return _LIKELY_MY_HEADER + + # If the target and include share some initial basename + # component, it's possible the target is implementing the + # include, so it's allowed to be first, but we'll never + # complain if it's not there. + target_first_component = _RE_FIRST_COMPONENT.match(target_base) + include_first_component = _RE_FIRST_COMPONENT.match(include_base) + if (target_first_component and include_first_component and + target_first_component.group(0) == + include_first_component.group(0)): + return _POSSIBLE_MY_HEADER + + return _OTHER_HEADER + + + +def CheckIncludeLine(filename, clean_lines, linenum, include_state, error): + """Check rules that are applicable to #include lines. + + Strings on #include lines are NOT removed from elided line, to make + certain tasks easier. However, to prevent false positives, checks + applicable to #include lines in CheckLanguage must be put here. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + include_state: An _IncludeState instance in which the headers are inserted. + error: The function to call with any errors found. + """ + fileinfo = FileInfo(filename) + line = clean_lines.lines[linenum] + + # "include" should use the new style "foo/bar.h" instead of just "bar.h" + # Only do this check if the included header follows google naming + # conventions. If not, assume that it's a 3rd party API that + # requires special include conventions. + # + # We also make an exception for Lua headers, which follow google + # naming convention but not the include convention. + match = Match(r'#include\s*"([^/]+\.h)"', line) + if match and not _THIRD_PARTY_HEADERS_PATTERN.match(match.group(1)): + error(filename, linenum, 'build/include_subdir', 4, + 'Include the directory when naming .h files') + + # we shouldn't include a file more than once. actually, there are a + # handful of instances where doing so is okay, but in general it's + # not. + match = _RE_PATTERN_INCLUDE.search(line) + if match: + include = match.group(2) + is_system = (match.group(1) == '<') + duplicate_line = include_state.FindHeader(include) + if duplicate_line >= 0: + error(filename, linenum, 'build/include', 4, + '"%s" already included at %s:%s' % + (include, filename, duplicate_line)) + return + + for extension in GetNonHeaderExtensions(): + if (include.endswith('.' + extension) and + os.path.dirname(fileinfo.RepositoryName()) != os.path.dirname(include)): + error(filename, linenum, 'build/include', 4, + 'Do not include .' + extension + ' files from other packages') + return + + if not _THIRD_PARTY_HEADERS_PATTERN.match(include): + include_state.include_list[-1].append((include, linenum)) + + # We want to ensure that headers appear in the right order: + # 1) for foo.cc, foo.h (preferred location) + # 2) c system files + # 3) cpp system files + # 4) for foo.cc, foo.h (deprecated location) + # 5) other google headers + # + # We classify each include statement as one of those 5 types + # using a number of techniques. The include_state object keeps + # track of the highest type seen, and complains if we see a + # lower type after that. + error_message = include_state.CheckNextIncludeOrder( + _ClassifyInclude(fileinfo, include, is_system)) + if error_message: + error(filename, linenum, 'build/include_order', 4, + '%s. Should be: %s.h, c system, c++ system, other.' % + (error_message, fileinfo.BaseName())) + canonical_include = include_state.CanonicalizeAlphabeticalOrder(include) + if not include_state.IsInAlphabeticalOrder( + clean_lines, linenum, canonical_include): + error(filename, linenum, 'build/include_alpha', 4, + 'Include "%s" not in alphabetical order' % include) + include_state.SetLastHeader(canonical_include) + + + +def _GetTextInside(text, start_pattern): + r"""Retrieves all the text between matching open and close parentheses. + + Given a string of lines and a regular expression string, retrieve all the text + following the expression and between opening punctuation symbols like + (, [, or {, and the matching close-punctuation symbol. This properly nested + occurrences of the punctuations, so for the text like + printf(a(), b(c())); + a call to _GetTextInside(text, r'printf\(') will return 'a(), b(c())'. + start_pattern must match string having an open punctuation symbol at the end. + + Args: + text: The lines to extract text. Its comments and strings must be elided. + It can be single line and can span multiple lines. + start_pattern: The regexp string indicating where to start extracting + the text. + Returns: + The extracted text. + None if either the opening string or ending punctuation could not be found. + """ + # TODO(unknown): Audit cpplint.py to see what places could be profitably + # rewritten to use _GetTextInside (and use inferior regexp matching today). + + # Give opening punctuations to get the matching close-punctuations. + matching_punctuation = {'(': ')', '{': '}', '[': ']'} + closing_punctuation = set(itervalues(matching_punctuation)) + + # Find the position to start extracting text. + match = re.search(start_pattern, text, re.M) + if not match: # start_pattern not found in text. + return None + start_position = match.end(0) + + assert start_position > 0, ( + 'start_pattern must ends with an opening punctuation.') + assert text[start_position - 1] in matching_punctuation, ( + 'start_pattern must ends with an opening punctuation.') + # Stack of closing punctuations we expect to have in text after position. + punctuation_stack = [matching_punctuation[text[start_position - 1]]] + position = start_position + while punctuation_stack and position < len(text): + if text[position] == punctuation_stack[-1]: + punctuation_stack.pop() + elif text[position] in closing_punctuation: + # A closing punctuation without matching opening punctuations. + return None + elif text[position] in matching_punctuation: + punctuation_stack.append(matching_punctuation[text[position]]) + position += 1 + if punctuation_stack: + # Opening punctuations left without matching close-punctuations. + return None + # punctuations match. + return text[start_position:position - 1] + + +# Patterns for matching call-by-reference parameters. +# +# Supports nested templates up to 2 levels deep using this messy pattern: +# < (?: < (?: < [^<>]* +# > +# | [^<>] )* +# > +# | [^<>] )* +# > +_RE_PATTERN_IDENT = r'[_a-zA-Z]\w*' # =~ [[:alpha:]][[:alnum:]]* +_RE_PATTERN_TYPE = ( + r'(?:const\s+)?(?:typename\s+|class\s+|struct\s+|union\s+|enum\s+)?' + r'(?:\w|' + r'\s*<(?:<(?:<[^<>]*>|[^<>])*>|[^<>])*>|' + r'::)+') +# A call-by-reference parameter ends with '& identifier'. +_RE_PATTERN_REF_PARAM = re.compile( + r'(' + _RE_PATTERN_TYPE + r'(?:\s*(?:\bconst\b|[*]))*\s*' + r'&\s*' + _RE_PATTERN_IDENT + r')\s*(?:=[^,()]+)?[,)]') +# A call-by-const-reference parameter either ends with 'const& identifier' +# or looks like 'const type& identifier' when 'type' is atomic. +_RE_PATTERN_CONST_REF_PARAM = ( + r'(?:.*\s*\bconst\s*&\s*' + _RE_PATTERN_IDENT + + r'|const\s+' + _RE_PATTERN_TYPE + r'\s*&\s*' + _RE_PATTERN_IDENT + r')') +# Stream types. +_RE_PATTERN_REF_STREAM_PARAM = ( + r'(?:.*stream\s*&\s*' + _RE_PATTERN_IDENT + r')') + + +def CheckLanguage(filename, clean_lines, linenum, file_extension, + include_state, nesting_state, error): + """Checks rules from the 'C++ language rules' section of cppguide.html. + + Some of these rules are hard to test (function overloading, using + uint32 inappropriately), but we do the best we can. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + file_extension: The extension (without the dot) of the filename. + include_state: An _IncludeState instance in which the headers are inserted. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: The function to call with any errors found. + """ + # If the line is empty or consists of entirely a comment, no need to + # check it. + line = clean_lines.elided[linenum] + if not line: + return + + match = _RE_PATTERN_INCLUDE.search(line) + if match: + CheckIncludeLine(filename, clean_lines, linenum, include_state, error) + return + + # Reset include state across preprocessor directives. This is meant + # to silence warnings for conditional includes. + match = Match(r'^\s*#\s*(if|ifdef|ifndef|elif|else|endif)\b', line) + if match: + include_state.ResetSection(match.group(1)) + + + # Perform other checks now that we are sure that this is not an include line + CheckCasts(filename, clean_lines, linenum, error) + CheckGlobalStatic(filename, clean_lines, linenum, error) + CheckPrintf(filename, clean_lines, linenum, error) + + if IsHeaderExtension(file_extension): + # TODO(unknown): check that 1-arg constructors are explicit. + # How to tell it's a constructor? + # (handled in CheckForNonStandardConstructs for now) + # TODO(unknown): check that classes declare or disable copy/assign + # (level 1 error) + pass + + # Check if people are using the verboten C basic types. The only exception + # we regularly allow is "unsigned short port" for port. + if Search(r'\bshort port\b', line): + if not Search(r'\bunsigned short port\b', line): + error(filename, linenum, 'runtime/int', 4, + 'Use "unsigned short" for ports, not "short"') + else: + match = Search(r'\b(short|long(?! +double)|long long)\b', line) + if match: + error(filename, linenum, 'runtime/int', 4, + 'Use int16/int64/etc, rather than the C type %s' % match.group(1)) + + # Check if some verboten operator overloading is going on + # TODO(unknown): catch out-of-line unary operator&: + # class X {}; + # int operator&(const X& x) { return 42; } // unary operator& + # The trick is it's hard to tell apart from binary operator&: + # class Y { int operator&(const Y& x) { return 23; } }; // binary operator& + if Search(r'\boperator\s*&\s*\(\s*\)', line): + error(filename, linenum, 'runtime/operator', 4, + 'Unary operator& is dangerous. Do not use it.') + + # Check for suspicious usage of "if" like + # } if (a == b) { + if Search(r'\}\s*if\s*\(', line): + error(filename, linenum, 'readability/braces', 4, + 'Did you mean "else if"? If not, start a new line for "if".') + + # Check for potential format string bugs like printf(foo). + # We constrain the pattern not to pick things like DocidForPrintf(foo). + # Not perfect but it can catch printf(foo.c_str()) and printf(foo->c_str()) + # TODO(unknown): Catch the following case. Need to change the calling + # convention of the whole function to process multiple line to handle it. + # printf( + # boy_this_is_a_really_long_variable_that_cannot_fit_on_the_prev_line); + printf_args = _GetTextInside(line, r'(?i)\b(string)?printf\s*\(') + if printf_args: + match = Match(r'([\w.\->()]+)$', printf_args) + if match and match.group(1) != '__VA_ARGS__': + function_name = re.search(r'\b((?:string)?printf)\s*\(', + line, re.I).group(1) + error(filename, linenum, 'runtime/printf', 4, + 'Potential format string bug. Do %s("%%s", %s) instead.' + % (function_name, match.group(1))) + + # Check for potential memset bugs like memset(buf, sizeof(buf), 0). + match = Search(r'memset\s*\(([^,]*),\s*([^,]*),\s*0\s*\)', line) + if match and not Match(r"^''|-?[0-9]+|0x[0-9A-Fa-f]$", match.group(2)): + error(filename, linenum, 'runtime/memset', 4, + 'Did you mean "memset(%s, 0, %s)"?' + % (match.group(1), match.group(2))) + + if Search(r'\busing namespace\b', line): + if Search(r'\bliterals\b', line): + error(filename, linenum, 'build/namespaces_literals', 5, + 'Do not use namespace using-directives. ' + 'Use using-declarations instead.') + else: + error(filename, linenum, 'build/namespaces', 5, + 'Do not use namespace using-directives. ' + 'Use using-declarations instead.') + + # Detect variable-length arrays. + match = Match(r'\s*(.+::)?(\w+) [a-z]\w*\[(.+)];', line) + if (match and match.group(2) != 'return' and match.group(2) != 'delete' and + match.group(3).find(']') == -1): + # Split the size using space and arithmetic operators as delimiters. + # If any of the resulting tokens are not compile time constants then + # report the error. + tokens = re.split(r'\s|\+|\-|\*|\/|<<|>>]', match.group(3)) + is_const = True + skip_next = False + for tok in tokens: + if skip_next: + skip_next = False + continue + + if Search(r'sizeof\(.+\)', tok): continue + if Search(r'arraysize\(\w+\)', tok): continue + + tok = tok.lstrip('(') + tok = tok.rstrip(')') + if not tok: continue + if Match(r'\d+', tok): continue + if Match(r'0[xX][0-9a-fA-F]+', tok): continue + if Match(r'k[A-Z0-9]\w*', tok): continue + if Match(r'(.+::)?k[A-Z0-9]\w*', tok): continue + if Match(r'(.+::)?[A-Z][A-Z0-9_]*', tok): continue + # A catch all for tricky sizeof cases, including 'sizeof expression', + # 'sizeof(*type)', 'sizeof(const type)', 'sizeof(struct StructName)' + # requires skipping the next token because we split on ' ' and '*'. + if tok.startswith('sizeof'): + skip_next = True + continue + is_const = False + break + if not is_const: + error(filename, linenum, 'runtime/arrays', 1, + 'Do not use variable-length arrays. Use an appropriately named ' + "('k' followed by CamelCase) compile-time constant for the size.") + + # Check for use of unnamed namespaces in header files. Registration + # macros are typically OK, so we allow use of "namespace {" on lines + # that end with backslashes. + if (IsHeaderExtension(file_extension) + and Search(r'\bnamespace\s*{', line) + and line[-1] != '\\'): + error(filename, linenum, 'build/namespaces', 4, + 'Do not use unnamed namespaces in header files. See ' + 'https://google-styleguide.googlecode.com/svn/trunk/cppguide.xml#Namespaces' + ' for more information.') + + +def CheckGlobalStatic(filename, clean_lines, linenum, error): + """Check for unsafe global or static objects. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Match two lines at a time to support multiline declarations + if linenum + 1 < clean_lines.NumLines() and not Search(r'[;({]', line): + line += clean_lines.elided[linenum + 1].strip() + + # Check for people declaring static/global STL strings at the top level. + # This is dangerous because the C++ language does not guarantee that + # globals with constructors are initialized before the first access, and + # also because globals can be destroyed when some threads are still running. + # TODO(unknown): Generalize this to also find static unique_ptr instances. + # TODO(unknown): File bugs for clang-tidy to find these. + match = Match( + r'((?:|static +)(?:|const +))(?::*std::)?string( +const)? +' + r'([a-zA-Z0-9_:]+)\b(.*)', + line) + + # Remove false positives: + # - String pointers (as opposed to values). + # string *pointer + # const string *pointer + # string const *pointer + # string *const pointer + # + # - Functions and template specializations. + # string Function(... + # string Class::Method(... + # + # - Operators. These are matched separately because operator names + # cross non-word boundaries, and trying to match both operators + # and functions at the same time would decrease accuracy of + # matching identifiers. + # string Class::operator*() + if (match and + not Search(r'\bstring\b(\s+const)?\s*[\*\&]\s*(const\s+)?\w', line) and + not Search(r'\boperator\W', line) and + not Match(r'\s*(<.*>)?(::[a-zA-Z0-9_]+)*\s*\(([^"]|$)', match.group(4))): + if Search(r'\bconst\b', line): + error(filename, linenum, 'runtime/string', 4, + 'For a static/global string constant, use a C style string ' + 'instead: "%schar%s %s[]".' % + (match.group(1), match.group(2) or '', match.group(3))) + else: + error(filename, linenum, 'runtime/string', 4, + 'Static/global string variables are not permitted.') + + if (Search(r'\b([A-Za-z0-9_]*_)\(\1\)', line) or + Search(r'\b([A-Za-z0-9_]*_)\(CHECK_NOTNULL\(\1\)\)', line)): + error(filename, linenum, 'runtime/init', 4, + 'You seem to be initializing a member variable with itself.') + + +def CheckPrintf(filename, clean_lines, linenum, error): + """Check for printf related issues. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # When snprintf is used, the second argument shouldn't be a literal. + match = Search(r'snprintf\s*\(([^,]*),\s*([0-9]*)\s*,', line) + if match and match.group(2) != '0': + # If 2nd arg is zero, snprintf is used to calculate size. + error(filename, linenum, 'runtime/printf', 3, + 'If you can, use sizeof(%s) instead of %s as the 2nd arg ' + 'to snprintf.' % (match.group(1), match.group(2))) + + # Check if some verboten C functions are being used. + if Search(r'\bsprintf\s*\(', line): + error(filename, linenum, 'runtime/printf', 5, + 'Never use sprintf. Use snprintf instead.') + match = Search(r'\b(strcpy|strcat)\s*\(', line) + if match: + error(filename, linenum, 'runtime/printf', 4, + 'Almost always, snprintf is better than %s' % match.group(1)) + + +def IsDerivedFunction(clean_lines, linenum): + """Check if current line contains an inherited function. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + Returns: + True if current line contains a function with "override" + virt-specifier. + """ + # Scan back a few lines for start of current function + for i in xrange(linenum, max(-1, linenum - 10), -1): + match = Match(r'^([^()]*\w+)\(', clean_lines.elided[i]) + if match: + # Look for "override" after the matching closing parenthesis + line, _, closing_paren = CloseExpression( + clean_lines, i, len(match.group(1))) + return (closing_paren >= 0 and + Search(r'\boverride\b', line[closing_paren:])) + return False + + +def IsOutOfLineMethodDefinition(clean_lines, linenum): + """Check if current line contains an out-of-line method definition. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + Returns: + True if current line contains an out-of-line method definition. + """ + # Scan back a few lines for start of current function + for i in xrange(linenum, max(-1, linenum - 10), -1): + if Match(r'^([^()]*\w+)\(', clean_lines.elided[i]): + return Match(r'^[^()]*\w+::\w+\(', clean_lines.elided[i]) is not None + return False + + +def IsInitializerList(clean_lines, linenum): + """Check if current line is inside constructor initializer list. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + Returns: + True if current line appears to be inside constructor initializer + list, False otherwise. + """ + for i in xrange(linenum, 1, -1): + line = clean_lines.elided[i] + if i == linenum: + remove_function_body = Match(r'^(.*)\{\s*$', line) + if remove_function_body: + line = remove_function_body.group(1) + + if Search(r'\s:\s*\w+[({]', line): + # A lone colon tend to indicate the start of a constructor + # initializer list. It could also be a ternary operator, which + # also tend to appear in constructor initializer lists as + # opposed to parameter lists. + return True + if Search(r'\}\s*,\s*$', line): + # A closing brace followed by a comma is probably the end of a + # brace-initialized member in constructor initializer list. + return True + if Search(r'[{};]\s*$', line): + # Found one of the following: + # - A closing brace or semicolon, probably the end of the previous + # function. + # - An opening brace, probably the start of current class or namespace. + # + # Current line is probably not inside an initializer list since + # we saw one of those things without seeing the starting colon. + return False + + # Got to the beginning of the file without seeing the start of + # constructor initializer list. + return False + + +def CheckForNonConstReference(filename, clean_lines, linenum, + nesting_state, error): + """Check for non-const references. + + Separate from CheckLanguage since it scans backwards from current + line, instead of scanning forward. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: The function to call with any errors found. + """ + # Do nothing if there is no '&' on current line. + line = clean_lines.elided[linenum] + if '&' not in line: + return + + # If a function is inherited, current function doesn't have much of + # a choice, so any non-const references should not be blamed on + # derived function. + if IsDerivedFunction(clean_lines, linenum): + return + + # Don't warn on out-of-line method definitions, as we would warn on the + # in-line declaration, if it isn't marked with 'override'. + if IsOutOfLineMethodDefinition(clean_lines, linenum): + return + + # Long type names may be broken across multiple lines, usually in one + # of these forms: + # LongType + # ::LongTypeContinued &identifier + # LongType:: + # LongTypeContinued &identifier + # LongType< + # ...>::LongTypeContinued &identifier + # + # If we detected a type split across two lines, join the previous + # line to current line so that we can match const references + # accordingly. + # + # Note that this only scans back one line, since scanning back + # arbitrary number of lines would be expensive. If you have a type + # that spans more than 2 lines, please use a typedef. + if linenum > 1: + previous = None + if Match(r'\s*::(?:[\w<>]|::)+\s*&\s*\S', line): + # previous_line\n + ::current_line + previous = Search(r'\b((?:const\s*)?(?:[\w<>]|::)+[\w<>])\s*$', + clean_lines.elided[linenum - 1]) + elif Match(r'\s*[a-zA-Z_]([\w<>]|::)+\s*&\s*\S', line): + # previous_line::\n + current_line + previous = Search(r'\b((?:const\s*)?(?:[\w<>]|::)+::)\s*$', + clean_lines.elided[linenum - 1]) + if previous: + line = previous.group(1) + line.lstrip() + else: + # Check for templated parameter that is split across multiple lines + endpos = line.rfind('>') + if endpos > -1: + (_, startline, startpos) = ReverseCloseExpression( + clean_lines, linenum, endpos) + if startpos > -1 and startline < linenum: + # Found the matching < on an earlier line, collect all + # pieces up to current line. + line = '' + for i in xrange(startline, linenum + 1): + line += clean_lines.elided[i].strip() + + # Check for non-const references in function parameters. A single '&' may + # found in the following places: + # inside expression: binary & for bitwise AND + # inside expression: unary & for taking the address of something + # inside declarators: reference parameter + # We will exclude the first two cases by checking that we are not inside a + # function body, including one that was just introduced by a trailing '{'. + # TODO(unknown): Doesn't account for 'catch(Exception& e)' [rare]. + if (nesting_state.previous_stack_top and + not (isinstance(nesting_state.previous_stack_top, _ClassInfo) or + isinstance(nesting_state.previous_stack_top, _NamespaceInfo))): + # Not at toplevel, not within a class, and not within a namespace + return + + # Avoid initializer lists. We only need to scan back from the + # current line for something that starts with ':'. + # + # We don't need to check the current line, since the '&' would + # appear inside the second set of parentheses on the current line as + # opposed to the first set. + if linenum > 0: + for i in xrange(linenum - 1, max(0, linenum - 10), -1): + previous_line = clean_lines.elided[i] + if not Search(r'[),]\s*$', previous_line): + break + if Match(r'^\s*:\s+\S', previous_line): + return + + # Avoid preprocessors + if Search(r'\\\s*$', line): + return + + # Avoid constructor initializer lists + if IsInitializerList(clean_lines, linenum): + return + + # We allow non-const references in a few standard places, like functions + # called "swap()" or iostream operators like "<<" or ">>". Do not check + # those function parameters. + # + # We also accept & in static_assert, which looks like a function but + # it's actually a declaration expression. + whitelisted_functions = (r'(?:[sS]wap(?:<\w:+>)?|' + r'operator\s*[<>][<>]|' + r'static_assert|COMPILE_ASSERT' + r')\s*\(') + if Search(whitelisted_functions, line): + return + elif not Search(r'\S+\([^)]*$', line): + # Don't see a whitelisted function on this line. Actually we + # didn't see any function name on this line, so this is likely a + # multi-line parameter list. Try a bit harder to catch this case. + for i in xrange(2): + if (linenum > i and + Search(whitelisted_functions, clean_lines.elided[linenum - i - 1])): + return + + decls = ReplaceAll(r'{[^}]*}', ' ', line) # exclude function body + for parameter in re.findall(_RE_PATTERN_REF_PARAM, decls): + if (not Match(_RE_PATTERN_CONST_REF_PARAM, parameter) and + not Match(_RE_PATTERN_REF_STREAM_PARAM, parameter)): + error(filename, linenum, 'runtime/references', 2, + 'Is this a non-const reference? ' + 'If so, make const or use a pointer: ' + + ReplaceAll(' *<', '<', parameter)) + + +def CheckCasts(filename, clean_lines, linenum, error): + """Various cast related checks. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + # Check to see if they're using an conversion function cast. + # I just try to capture the most common basic types, though there are more. + # Parameterless conversion functions, such as bool(), are allowed as they are + # probably a member operator declaration or default constructor. + match = Search( + r'(\bnew\s+(?:const\s+)?|\S<\s*(?:const\s+)?)?\b' + r'(int|float|double|bool|char|int32|uint32|int64|uint64)' + r'(\([^)].*)', line) + expecting_function = ExpectingFunctionArgs(clean_lines, linenum) + if match and not expecting_function: + matched_type = match.group(2) + + # matched_new_or_template is used to silence two false positives: + # - New operators + # - Template arguments with function types + # + # For template arguments, we match on types immediately following + # an opening bracket without any spaces. This is a fast way to + # silence the common case where the function type is the first + # template argument. False negative with less-than comparison is + # avoided because those operators are usually followed by a space. + # + # function // bracket + no space = false positive + # value < double(42) // bracket + space = true positive + matched_new_or_template = match.group(1) + + # Avoid arrays by looking for brackets that come after the closing + # parenthesis. + if Match(r'\([^()]+\)\s*\[', match.group(3)): + return + + # Other things to ignore: + # - Function pointers + # - Casts to pointer types + # - Placement new + # - Alias declarations + matched_funcptr = match.group(3) + if (matched_new_or_template is None and + not (matched_funcptr and + (Match(r'\((?:[^() ]+::\s*\*\s*)?[^() ]+\)\s*\(', + matched_funcptr) or + matched_funcptr.startswith('(*)'))) and + not Match(r'\s*using\s+\S+\s*=\s*' + matched_type, line) and + not Search(r'new\(\S+\)\s*' + matched_type, line)): + error(filename, linenum, 'readability/casting', 4, + 'Using deprecated casting style. ' + 'Use static_cast<%s>(...) instead' % + matched_type) + + if not expecting_function: + CheckCStyleCast(filename, clean_lines, linenum, 'static_cast', + r'\((int|float|double|bool|char|u?int(16|32|64))\)', error) + + # This doesn't catch all cases. Consider (const char * const)"hello". + # + # (char *) "foo" should always be a const_cast (reinterpret_cast won't + # compile). + if CheckCStyleCast(filename, clean_lines, linenum, 'const_cast', + r'\((char\s?\*+\s?)\)\s*"', error): + pass + else: + # Check pointer casts for other than string constants + CheckCStyleCast(filename, clean_lines, linenum, 'reinterpret_cast', + r'\((\w+\s?\*+\s?)\)', error) + + # In addition, we look for people taking the address of a cast. This + # is dangerous -- casts can assign to temporaries, so the pointer doesn't + # point where you think. + # + # Some non-identifier character is required before the '&' for the + # expression to be recognized as a cast. These are casts: + # expression = &static_cast(temporary()); + # function(&(int*)(temporary())); + # + # This is not a cast: + # reference_type&(int* function_param); + match = Search( + r'(?:[^\w]&\(([^)*][^)]*)\)[\w(])|' + r'(?:[^\w]&(static|dynamic|down|reinterpret)_cast\b)', line) + if match: + # Try a better error message when the & is bound to something + # dereferenced by the casted pointer, as opposed to the casted + # pointer itself. + parenthesis_error = False + match = Match(r'^(.*&(?:static|dynamic|down|reinterpret)_cast\b)<', line) + if match: + _, y1, x1 = CloseExpression(clean_lines, linenum, len(match.group(1))) + if x1 >= 0 and clean_lines.elided[y1][x1] == '(': + _, y2, x2 = CloseExpression(clean_lines, y1, x1) + if x2 >= 0: + extended_line = clean_lines.elided[y2][x2:] + if y2 < clean_lines.NumLines() - 1: + extended_line += clean_lines.elided[y2 + 1] + if Match(r'\s*(?:->|\[)', extended_line): + parenthesis_error = True + + if parenthesis_error: + error(filename, linenum, 'readability/casting', 4, + ('Are you taking an address of something dereferenced ' + 'from a cast? Wrapping the dereferenced expression in ' + 'parentheses will make the binding more obvious')) + else: + error(filename, linenum, 'runtime/casting', 4, + ('Are you taking an address of a cast? ' + 'This is dangerous: could be a temp var. ' + 'Take the address before doing the cast, rather than after')) + + +def CheckCStyleCast(filename, clean_lines, linenum, cast_type, pattern, error): + """Checks for a C-style cast by looking for the pattern. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + cast_type: The string for the C++ cast to recommend. This is either + reinterpret_cast, static_cast, or const_cast, depending. + pattern: The regular expression used to find C-style casts. + error: The function to call with any errors found. + + Returns: + True if an error was emitted. + False otherwise. + """ + line = clean_lines.elided[linenum] + match = Search(pattern, line) + if not match: + return False + + # Exclude lines with keywords that tend to look like casts + context = line[0:match.start(1) - 1] + if Match(r'.*\b(?:sizeof|alignof|alignas|[_A-Z][_A-Z0-9]*)\s*$', context): + return False + + # Try expanding current context to see if we one level of + # parentheses inside a macro. + if linenum > 0: + for i in xrange(linenum - 1, max(0, linenum - 5), -1): + context = clean_lines.elided[i] + context + if Match(r'.*\b[_A-Z][_A-Z0-9]*\s*\((?:\([^()]*\)|[^()])*$', context): + return False + + # operator++(int) and operator--(int) + if context.endswith(' operator++') or context.endswith(' operator--'): + return False + + # A single unnamed argument for a function tends to look like old style cast. + # If we see those, don't issue warnings for deprecated casts. + remainder = line[match.end(0):] + if Match(r'^\s*(?:;|const\b|throw\b|final\b|override\b|[=>{),]|->)', + remainder): + return False + + # At this point, all that should be left is actual casts. + error(filename, linenum, 'readability/casting', 4, + 'Using C-style cast. Use %s<%s>(...) instead' % + (cast_type, match.group(1))) + + return True + + +def ExpectingFunctionArgs(clean_lines, linenum): + """Checks whether where function type arguments are expected. + + Args: + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + + Returns: + True if the line at 'linenum' is inside something that expects arguments + of function types. + """ + line = clean_lines.elided[linenum] + return (Match(r'^\s*MOCK_(CONST_)?METHOD\d+(_T)?\(', line) or + (linenum >= 2 and + (Match(r'^\s*MOCK_(?:CONST_)?METHOD\d+(?:_T)?\((?:\S+,)?\s*$', + clean_lines.elided[linenum - 1]) or + Match(r'^\s*MOCK_(?:CONST_)?METHOD\d+(?:_T)?\(\s*$', + clean_lines.elided[linenum - 2]) or + Search(r'\bstd::m?function\s*\<\s*$', + clean_lines.elided[linenum - 1])))) + + +_HEADERS_CONTAINING_TEMPLATES = ( + ('', ('deque',)), + ('', ('unary_function', 'binary_function', + 'plus', 'minus', 'multiplies', 'divides', 'modulus', + 'negate', + 'equal_to', 'not_equal_to', 'greater', 'less', + 'greater_equal', 'less_equal', + 'logical_and', 'logical_or', 'logical_not', + 'unary_negate', 'not1', 'binary_negate', 'not2', + 'bind1st', 'bind2nd', + 'pointer_to_unary_function', + 'pointer_to_binary_function', + 'ptr_fun', + 'mem_fun_t', 'mem_fun', 'mem_fun1_t', 'mem_fun1_ref_t', + 'mem_fun_ref_t', + 'const_mem_fun_t', 'const_mem_fun1_t', + 'const_mem_fun_ref_t', 'const_mem_fun1_ref_t', + 'mem_fun_ref', + )), + ('', ('numeric_limits',)), + ('', ('list',)), + ('', ('map', 'multimap',)), + ('', ('allocator', 'make_shared', 'make_unique', 'shared_ptr', + 'unique_ptr', 'weak_ptr')), + ('', ('queue', 'priority_queue',)), + ('', ('set', 'multiset',)), + ('', ('stack',)), + ('', ('char_traits', 'basic_string',)), + ('', ('tuple',)), + ('', ('unordered_map', 'unordered_multimap')), + ('', ('unordered_set', 'unordered_multiset')), + ('', ('pair',)), + ('', ('vector',)), + + # gcc extensions. + # Note: std::hash is their hash, ::hash is our hash + ('', ('hash_map', 'hash_multimap',)), + ('', ('hash_set', 'hash_multiset',)), + ('', ('slist',)), + ) + +_HEADERS_MAYBE_TEMPLATES = ( + ('', ('copy', 'max', 'min', 'min_element', 'sort', + 'transform', + )), + ('', ('forward', 'make_pair', 'move', 'swap')), + ) + +_RE_PATTERN_STRING = re.compile(r'\bstring\b') + +_re_pattern_headers_maybe_templates = [] +for _header, _templates in _HEADERS_MAYBE_TEMPLATES: + for _template in _templates: + # Match max(..., ...), max(..., ...), but not foo->max, foo.max or + # 'type::max()'. + _re_pattern_headers_maybe_templates.append( + (re.compile(r'[^>.]\b' + _template + r'(<.*?>)?\([^\)]'), + _template, + _header)) + +# Other scripts may reach in and modify this pattern. +_re_pattern_templates = [] +for _header, _templates in _HEADERS_CONTAINING_TEMPLATES: + for _template in _templates: + _re_pattern_templates.append( + (re.compile(r'(\<|\b)' + _template + r'\s*\<'), + _template + '<>', + _header)) + + +def FilesBelongToSameModule(filename_cc, filename_h): + """Check if these two filenames belong to the same module. + + The concept of a 'module' here is a as follows: + foo.h, foo-inl.h, foo.cc, foo_test.cc and foo_unittest.cc belong to the + same 'module' if they are in the same directory. + some/path/public/xyzzy and some/path/internal/xyzzy are also considered + to belong to the same module here. + + If the filename_cc contains a longer path than the filename_h, for example, + '/absolute/path/to/base/sysinfo.cc', and this file would include + 'base/sysinfo.h', this function also produces the prefix needed to open the + header. This is used by the caller of this function to more robustly open the + header file. We don't have access to the real include paths in this context, + so we need this guesswork here. + + Known bugs: tools/base/bar.cc and base/bar.h belong to the same module + according to this implementation. Because of this, this function gives + some false positives. This should be sufficiently rare in practice. + + Args: + filename_cc: is the path for the source (e.g. .cc) file + filename_h: is the path for the header path + + Returns: + Tuple with a bool and a string: + bool: True if filename_cc and filename_h belong to the same module. + string: the additional prefix needed to open the header file. + """ + fileinfo_cc = FileInfo(filename_cc) + if not fileinfo_cc.Extension().lstrip('.') in GetNonHeaderExtensions(): + return (False, '') + + fileinfo_h = FileInfo(filename_h) + if not IsHeaderExtension(fileinfo_h.Extension().lstrip('.')): + return (False, '') + + filename_cc = filename_cc[:-(len(fileinfo_cc.Extension()))] + matched_test_suffix = Search(_TEST_FILE_SUFFIX, fileinfo_cc.BaseName()) + if matched_test_suffix: + filename_cc = filename_cc[:-len(matched_test_suffix.group(1))] + + filename_cc = filename_cc.replace('/public/', '/') + filename_cc = filename_cc.replace('/internal/', '/') + + filename_h = filename_h[:-(len(fileinfo_h.Extension()))] + if filename_h.endswith('-inl'): + filename_h = filename_h[:-len('-inl')] + filename_h = filename_h.replace('/public/', '/') + filename_h = filename_h.replace('/internal/', '/') + + files_belong_to_same_module = filename_cc.endswith(filename_h) + common_path = '' + if files_belong_to_same_module: + common_path = filename_cc[:-len(filename_h)] + return files_belong_to_same_module, common_path + + +def UpdateIncludeState(filename, include_dict, io=codecs): + """Fill up the include_dict with new includes found from the file. + + Args: + filename: the name of the header to read. + include_dict: a dictionary in which the headers are inserted. + io: The io factory to use to read the file. Provided for testability. + + Returns: + True if a header was successfully added. False otherwise. + """ + headerfile = None + try: + headerfile = io.open(filename, 'r', 'utf8', 'replace') + except IOError: + return False + linenum = 0 + for line in headerfile: + linenum += 1 + clean_line = CleanseComments(line) + match = _RE_PATTERN_INCLUDE.search(clean_line) + if match: + include = match.group(2) + include_dict.setdefault(include, linenum) + return True + + +def CheckForIncludeWhatYouUse(filename, clean_lines, include_state, error, + io=codecs): + """Reports for missing stl includes. + + This function will output warnings to make sure you are including the headers + necessary for the stl containers and functions that you use. We only give one + reason to include a header. For example, if you use both equal_to<> and + less<> in a .h file, only one (the latter in the file) of these will be + reported as a reason to include the . + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + include_state: An _IncludeState instance. + error: The function to call with any errors found. + io: The IO factory to use to read the header file. Provided for unittest + injection. + """ + required = {} # A map of header name to linenumber and the template entity. + # Example of required: { '': (1219, 'less<>') } + + for linenum in xrange(clean_lines.NumLines()): + line = clean_lines.elided[linenum] + if not line or line[0] == '#': + continue + + # String is special -- it is a non-templatized type in STL. + matched = _RE_PATTERN_STRING.search(line) + if matched: + # Don't warn about strings in non-STL namespaces: + # (We check only the first match per line; good enough.) + prefix = line[:matched.start()] + if prefix.endswith('std::') or not prefix.endswith('::'): + required[''] = (linenum, 'string') + + for pattern, template, header in _re_pattern_headers_maybe_templates: + if pattern.search(line): + required[header] = (linenum, template) + + # The following function is just a speed up, no semantics are changed. + if not '<' in line: # Reduces the cpu time usage by skipping lines. + continue + + for pattern, template, header in _re_pattern_templates: + matched = pattern.search(line) + if matched: + # Don't warn about IWYU in non-STL namespaces: + # (We check only the first match per line; good enough.) + prefix = line[:matched.start()] + if prefix.endswith('std::') or not prefix.endswith('::'): + required[header] = (linenum, template) + + # The policy is that if you #include something in foo.h you don't need to + # include it again in foo.cc. Here, we will look at possible includes. + # Let's flatten the include_state include_list and copy it into a dictionary. + include_dict = dict([item for sublist in include_state.include_list + for item in sublist]) + + # Did we find the header for this file (if any) and successfully load it? + header_found = False + + # Use the absolute path so that matching works properly. + abs_filename = FileInfo(filename).FullName() + + # For Emacs's flymake. + # If cpplint is invoked from Emacs's flymake, a temporary file is generated + # by flymake and that file name might end with '_flymake.cc'. In that case, + # restore original file name here so that the corresponding header file can be + # found. + # e.g. If the file name is 'foo_flymake.cc', we should search for 'foo.h' + # instead of 'foo_flymake.h' + abs_filename = re.sub(r'_flymake\.cc$', '.cc', abs_filename) + + # include_dict is modified during iteration, so we iterate over a copy of + # the keys. + header_keys = list(include_dict.keys()) + for header in header_keys: + (same_module, common_path) = FilesBelongToSameModule(abs_filename, header) + fullpath = common_path + header + if same_module and UpdateIncludeState(fullpath, include_dict, io): + header_found = True + + # If we can't find the header file for a .cc, assume it's because we don't + # know where to look. In that case we'll give up as we're not sure they + # didn't include it in the .h file. + # TODO(unknown): Do a better job of finding .h files so we are confident that + # not having the .h file means there isn't one. + if not header_found: + for extension in GetNonHeaderExtensions(): + if filename.endswith('.' + extension): + return + + # All the lines have been processed, report the errors found. + for required_header_unstripped in sorted(required, key=required.__getitem__): + template = required[required_header_unstripped][1] + if required_header_unstripped.strip('<>"') not in include_dict: + error(filename, required[required_header_unstripped][0], + 'build/include_what_you_use', 4, + 'Add #include ' + required_header_unstripped + ' for ' + template) + + +_RE_PATTERN_EXPLICIT_MAKEPAIR = re.compile(r'\bmake_pair\s*<') + + +def CheckMakePairUsesDeduction(filename, clean_lines, linenum, error): + """Check that make_pair's template arguments are deduced. + + G++ 4.6 in C++11 mode fails badly if make_pair's template arguments are + specified explicitly, and such use isn't intended in any case. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + match = _RE_PATTERN_EXPLICIT_MAKEPAIR.search(line) + if match: + error(filename, linenum, 'build/explicit_make_pair', + 4, # 4 = high confidence + 'For C++11-compatibility, omit template arguments from make_pair' + ' OR use pair directly OR if appropriate, construct a pair directly') + + +def CheckRedundantVirtual(filename, clean_lines, linenum, error): + """Check if line contains a redundant "virtual" function-specifier. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + # Look for "virtual" on current line. + line = clean_lines.elided[linenum] + virtual = Match(r'^(.*)(\bvirtual\b)(.*)$', line) + if not virtual: return + + # Ignore "virtual" keywords that are near access-specifiers. These + # are only used in class base-specifier and do not apply to member + # functions. + if (Search(r'\b(public|protected|private)\s+$', virtual.group(1)) or + Match(r'^\s+(public|protected|private)\b', virtual.group(3))): + return + + # Ignore the "virtual" keyword from virtual base classes. Usually + # there is a column on the same line in these cases (virtual base + # classes are rare in google3 because multiple inheritance is rare). + if Match(r'^.*[^:]:[^:].*$', line): return + + # Look for the next opening parenthesis. This is the start of the + # parameter list (possibly on the next line shortly after virtual). + # TODO(unknown): doesn't work if there are virtual functions with + # decltype() or other things that use parentheses, but csearch suggests + # that this is rare. + end_col = -1 + end_line = -1 + start_col = len(virtual.group(2)) + for start_line in xrange(linenum, min(linenum + 3, clean_lines.NumLines())): + line = clean_lines.elided[start_line][start_col:] + parameter_list = Match(r'^([^(]*)\(', line) + if parameter_list: + # Match parentheses to find the end of the parameter list + (_, end_line, end_col) = CloseExpression( + clean_lines, start_line, start_col + len(parameter_list.group(1))) + break + start_col = 0 + + if end_col < 0: + return # Couldn't find end of parameter list, give up + + # Look for "override" or "final" after the parameter list + # (possibly on the next few lines). + for i in xrange(end_line, min(end_line + 3, clean_lines.NumLines())): + line = clean_lines.elided[i][end_col:] + match = Search(r'\b(override|final)\b', line) + if match: + error(filename, linenum, 'readability/inheritance', 4, + ('"virtual" is redundant since function is ' + 'already declared as "%s"' % match.group(1))) + + # Set end_col to check whole lines after we are done with the + # first line. + end_col = 0 + if Search(r'[^\w]\s*$', line): + break + + +def CheckRedundantOverrideOrFinal(filename, clean_lines, linenum, error): + """Check if line contains a redundant "override" or "final" virt-specifier. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + # Look for closing parenthesis nearby. We need one to confirm where + # the declarator ends and where the virt-specifier starts to avoid + # false positives. + line = clean_lines.elided[linenum] + declarator_end = line.rfind(')') + if declarator_end >= 0: + fragment = line[declarator_end:] + else: + if linenum > 1 and clean_lines.elided[linenum - 1].rfind(')') >= 0: + fragment = line + else: + return + + # Check that at most one of "override" or "final" is present, not both + if Search(r'\boverride\b', fragment) and Search(r'\bfinal\b', fragment): + error(filename, linenum, 'readability/inheritance', 4, + ('"override" is redundant since function is ' + 'already declared as "final"')) + + + + +# Returns true if we are at a new block, and it is directly +# inside of a namespace. +def IsBlockInNameSpace(nesting_state, is_forward_declaration): + """Checks that the new block is directly in a namespace. + + Args: + nesting_state: The _NestingState object that contains info about our state. + is_forward_declaration: If the class is a forward declared class. + Returns: + Whether or not the new block is directly in a namespace. + """ + if is_forward_declaration: + return len(nesting_state.stack) >= 1 and ( + isinstance(nesting_state.stack[-1], _NamespaceInfo)) + + + return (len(nesting_state.stack) > 1 and + nesting_state.stack[-1].check_namespace_indentation and + isinstance(nesting_state.stack[-2], _NamespaceInfo)) + + +def ShouldCheckNamespaceIndentation(nesting_state, is_namespace_indent_item, + raw_lines_no_comments, linenum): + """This method determines if we should apply our namespace indentation check. + + Args: + nesting_state: The current nesting state. + is_namespace_indent_item: If we just put a new class on the stack, True. + If the top of the stack is not a class, or we did not recently + add the class, False. + raw_lines_no_comments: The lines without the comments. + linenum: The current line number we are processing. + + Returns: + True if we should apply our namespace indentation check. Currently, it + only works for classes and namespaces inside of a namespace. + """ + + is_forward_declaration = IsForwardClassDeclaration(raw_lines_no_comments, + linenum) + + if not (is_namespace_indent_item or is_forward_declaration): + return False + + # If we are in a macro, we do not want to check the namespace indentation. + if IsMacroDefinition(raw_lines_no_comments, linenum): + return False + + return IsBlockInNameSpace(nesting_state, is_forward_declaration) + + +# Call this method if the line is directly inside of a namespace. +# If the line above is blank (excluding comments) or the start of +# an inner namespace, it cannot be indented. +def CheckItemIndentationInNamespace(filename, raw_lines_no_comments, linenum, + error): + line = raw_lines_no_comments[linenum] + if Match(r'^\s+', line): + error(filename, linenum, 'runtime/indentation_namespace', 4, + 'Do not indent within a namespace') + + +def ProcessLine(filename, file_extension, clean_lines, line, + include_state, function_state, nesting_state, error, + extra_check_functions=None): + """Processes a single line in the file. + + Args: + filename: Filename of the file that is being processed. + file_extension: The extension (dot not included) of the file. + clean_lines: An array of strings, each representing a line of the file, + with comments stripped. + line: Number of line being processed. + include_state: An _IncludeState instance in which the headers are inserted. + function_state: A _FunctionState instance which counts function lines, etc. + nesting_state: A NestingState instance which maintains information about + the current stack of nested blocks being parsed. + error: A callable to which errors are reported, which takes 4 arguments: + filename, line number, error level, and message + extra_check_functions: An array of additional check functions that will be + run on each source line. Each function takes 4 + arguments: filename, clean_lines, line, error + """ + raw_lines = clean_lines.raw_lines + ParseNolintSuppressions(filename, raw_lines[line], line, error) + nesting_state.Update(filename, clean_lines, line, error) + CheckForNamespaceIndentation(filename, nesting_state, clean_lines, line, + error) + if nesting_state.InAsmBlock(): return + CheckForFunctionLengths(filename, clean_lines, line, function_state, error) + CheckForMultilineCommentsAndStrings(filename, clean_lines, line, error) + CheckStyle(filename, clean_lines, line, file_extension, nesting_state, error) + CheckLanguage(filename, clean_lines, line, file_extension, include_state, + nesting_state, error) + CheckForNonConstReference(filename, clean_lines, line, nesting_state, error) + CheckForNonStandardConstructs(filename, clean_lines, line, + nesting_state, error) + CheckVlogArguments(filename, clean_lines, line, error) + CheckPosixThreading(filename, clean_lines, line, error) + CheckInvalidIncrement(filename, clean_lines, line, error) + CheckMakePairUsesDeduction(filename, clean_lines, line, error) + CheckRedundantVirtual(filename, clean_lines, line, error) + CheckRedundantOverrideOrFinal(filename, clean_lines, line, error) + if extra_check_functions: + for check_fn in extra_check_functions: + check_fn(filename, clean_lines, line, error) + +def FlagCxx11Features(filename, clean_lines, linenum, error): + """Flag those c++11 features that we only allow in certain places. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + include = Match(r'\s*#\s*include\s+[<"]([^<"]+)[">]', line) + + # Flag unapproved C++ TR1 headers. + if include and include.group(1).startswith('tr1/'): + error(filename, linenum, 'build/c++tr1', 5, + ('C++ TR1 headers such as <%s> are unapproved.') % include.group(1)) + + # Flag unapproved C++11 headers. + if include and include.group(1) in ('cfenv', + 'condition_variable', + 'fenv.h', + 'future', + 'mutex', + 'thread', + 'chrono', + 'ratio', + 'regex', + 'system_error', + ): + error(filename, linenum, 'build/c++11', 5, + ('<%s> is an unapproved C++11 header.') % include.group(1)) + + # The only place where we need to worry about C++11 keywords and library + # features in preprocessor directives is in macro definitions. + if Match(r'\s*#', line) and not Match(r'\s*#\s*define\b', line): return + + # These are classes and free functions. The classes are always + # mentioned as std::*, but we only catch the free functions if + # they're not found by ADL. They're alphabetical by header. + for top_name in ( + # type_traits + 'alignment_of', + 'aligned_union', + ): + if Search(r'\bstd::%s\b' % top_name, line): + error(filename, linenum, 'build/c++11', 5, + ('std::%s is an unapproved C++11 class or function. Send c-style ' + 'an example of where it would make your code more readable, and ' + 'they may let you use it.') % top_name) + + +def FlagCxx14Features(filename, clean_lines, linenum, error): + """Flag those C++14 features that we restrict. + + Args: + filename: The name of the current file. + clean_lines: A CleansedLines instance containing the file. + linenum: The number of the line to check. + error: The function to call with any errors found. + """ + line = clean_lines.elided[linenum] + + include = Match(r'\s*#\s*include\s+[<"]([^<"]+)[">]', line) + + # Flag unapproved C++14 headers. + if include and include.group(1) in ('scoped_allocator', 'shared_mutex'): + error(filename, linenum, 'build/c++14', 5, + ('<%s> is an unapproved C++14 header.') % include.group(1)) + + +def ProcessFileData(filename, file_extension, lines, error, + extra_check_functions=None): + """Performs lint checks and reports any errors to the given error function. + + Args: + filename: Filename of the file that is being processed. + file_extension: The extension (dot not included) of the file. + lines: An array of strings, each representing a line of the file, with the + last element being empty if the file is terminated with a newline. + error: A callable to which errors are reported, which takes 4 arguments: + filename, line number, error level, and message + extra_check_functions: An array of additional check functions that will be + run on each source line. Each function takes 4 + arguments: filename, clean_lines, line, error + """ + lines = (['// marker so line numbers and indices both start at 1'] + lines + + ['// marker so line numbers end in a known way']) + + include_state = _IncludeState() + function_state = _FunctionState() + nesting_state = NestingState() + + ResetNolintSuppressions() + + CheckForCopyright(filename, lines, error) + ProcessGlobalSuppresions(lines) + RemoveMultiLineComments(filename, lines, error) + clean_lines = CleansedLines(lines) + + if IsHeaderExtension(file_extension): + CheckForHeaderGuard(filename, clean_lines, error) + + for line in xrange(clean_lines.NumLines()): + ProcessLine(filename, file_extension, clean_lines, line, + include_state, function_state, nesting_state, error, + extra_check_functions) + FlagCxx11Features(filename, clean_lines, line, error) + nesting_state.CheckCompletedBlocks(filename, error) + + CheckForIncludeWhatYouUse(filename, clean_lines, include_state, error) + + # Check that the .cc file has included its header if it exists. + if _IsSourceExtension(file_extension): + CheckHeaderFileIncluded(filename, include_state, error) + + # We check here rather than inside ProcessLine so that we see raw + # lines rather than "cleaned" lines. + CheckForBadCharacters(filename, lines, error) + + CheckForNewlineAtEOF(filename, lines, error) + +def ProcessConfigOverrides(filename): + """ Loads the configuration files and processes the config overrides. + + Args: + filename: The name of the file being processed by the linter. + + Returns: + False if the current |filename| should not be processed further. + """ + + abs_filename = os.path.abspath(filename) + cfg_filters = [] + keep_looking = True + while keep_looking: + abs_path, base_name = os.path.split(abs_filename) + if not base_name: + break # Reached the root directory. + + cfg_file = os.path.join(abs_path, "CPPLINT.cfg") + abs_filename = abs_path + if not os.path.isfile(cfg_file): + continue + + try: + with open(cfg_file) as file_handle: + for line in file_handle: + line, _, _ = line.partition('#') # Remove comments. + if not line.strip(): + continue + + name, _, val = line.partition('=') + name = name.strip() + val = val.strip() + if name == 'set noparent': + keep_looking = False + elif name == 'filter': + cfg_filters.append(val) + elif name == 'exclude_files': + # When matching exclude_files pattern, use the base_name of + # the current file name or the directory name we are processing. + # For example, if we are checking for lint errors in /foo/bar/baz.cc + # and we found the .cfg file at /foo/CPPLINT.cfg, then the config + # file's "exclude_files" filter is meant to be checked against "bar" + # and not "baz" nor "bar/baz.cc". + if base_name: + pattern = re.compile(val) + if pattern.match(base_name): + if _cpplint_state.quiet: + # Suppress "Ignoring file" warning when using --quiet. + return False + _cpplint_state.PrintInfo('Ignoring "%s": file excluded by "%s". ' + 'File path component "%s" matches ' + 'pattern "%s"\n' % + (filename, cfg_file, base_name, val)) + return False + elif name == 'linelength': + global _line_length + try: + _line_length = int(val) + except ValueError: + _cpplint_state.PrintError('Line length must be numeric.') + elif name == 'extensions': + global _valid_extensions + try: + extensions = [ext.strip() for ext in val.split(',')] + _valid_extensions = set(extensions) + except ValueError: + sys.stderr.write('Extensions should be a comma-separated list of values;' + 'for example: extensions=hpp,cpp\n' + 'This could not be parsed: "%s"' % (val,)) + elif name == 'root': + global _root + # root directories are specified relative to CPPLINT.cfg dir. + _root = os.path.join(os.path.dirname(cfg_file), val) + elif name == 'headers': + ProcessHppHeadersOption(val) + else: + _cpplint_state.PrintError( + 'Invalid configuration option (%s) in file %s\n' % + (name, cfg_file)) + + except IOError: + _cpplint_state.PrintError( + "Skipping config file '%s': Can't open for reading\n" % cfg_file) + keep_looking = False + + # Apply all the accumulated filters in reverse order (top-level directory + # config options having the least priority). + for cfg_filter in reversed(cfg_filters): + _AddFilters(cfg_filter) + + return True + + +def ProcessFile(filename, vlevel, extra_check_functions=None): + """Does google-lint on a single file. + + Args: + filename: The name of the file to parse. + + vlevel: The level of errors to report. Every error of confidence + >= verbose_level will be reported. 0 is a good default. + + extra_check_functions: An array of additional check functions that will be + run on each source line. Each function takes 4 + arguments: filename, clean_lines, line, error + """ + + _SetVerboseLevel(vlevel) + _BackupFilters() + old_errors = _cpplint_state.error_count + + if not ProcessConfigOverrides(filename): + _RestoreFilters() + return + + lf_lines = [] + crlf_lines = [] + try: + # Support the UNIX convention of using "-" for stdin. Note that + # we are not opening the file with universal newline support + # (which codecs doesn't support anyway), so the resulting lines do + # contain trailing '\r' characters if we are reading a file that + # has CRLF endings. + # If after the split a trailing '\r' is present, it is removed + # below. + if filename == '-': + lines = codecs.StreamReaderWriter(sys.stdin, + codecs.getreader('utf8'), + codecs.getwriter('utf8'), + 'replace').read().split('\n') + else: + lines = codecs.open(filename, 'r', 'utf8', 'replace').read().split('\n') + + # Remove trailing '\r'. + # The -1 accounts for the extra trailing blank line we get from split() + for linenum in range(len(lines) - 1): + if lines[linenum].endswith('\r'): + lines[linenum] = lines[linenum].rstrip('\r') + crlf_lines.append(linenum + 1) + else: + lf_lines.append(linenum + 1) + + except IOError: + _cpplint_state.PrintError( + "Skipping input '%s': Can't open for reading\n" % filename) + _RestoreFilters() + return + + # Note, if no dot is found, this will give the entire filename as the ext. + file_extension = filename[filename.rfind('.') + 1:] + + # When reading from stdin, the extension is unknown, so no cpplint tests + # should rely on the extension. + if filename != '-' and file_extension not in GetAllExtensions(): + _cpplint_state.PrintError('Ignoring %s; not a valid file name ' + '(%s)\n' % (filename, ', '.join(GetAllExtensions()))) + else: + ProcessFileData(filename, file_extension, lines, Error, + extra_check_functions) + + # If end-of-line sequences are a mix of LF and CR-LF, issue + # warnings on the lines with CR. + # + # Don't issue any warnings if all lines are uniformly LF or CR-LF, + # since critique can handle these just fine, and the style guide + # doesn't dictate a particular end of line sequence. + # + # We can't depend on os.linesep to determine what the desired + # end-of-line sequence should be, since that will return the + # server-side end-of-line sequence. + if lf_lines and crlf_lines: + # Warn on every line with CR. An alternative approach might be to + # check whether the file is mostly CRLF or just LF, and warn on the + # minority, we bias toward LF here since most tools prefer LF. + for linenum in crlf_lines: + Error(filename, linenum, 'whitespace/newline', 1, + 'Unexpected \\r (^M) found; better to use only \\n') + + # Suppress printing anything if --quiet was passed unless the error + # count has increased after processing this file. + if not _cpplint_state.quiet or old_errors != _cpplint_state.error_count: + _cpplint_state.PrintInfo('Done processing %s\n' % filename) + _RestoreFilters() + + +def PrintUsage(message): + """Prints a brief usage string and exits, optionally with an error message. + + Args: + message: The optional error message. + """ + sys.stderr.write(_USAGE % (list(GetAllExtensions()), + ','.join(list(GetAllExtensions())), + GetHeaderExtensions(), + ','.join(GetHeaderExtensions()))) + + if message: + sys.exit('\nFATAL ERROR: ' + message) + else: + sys.exit(0) + +def PrintVersion(): + sys.stdout.write('Cpplint fork (https://github.com/cpplint/cpplint)\n') + sys.stdout.write('cpplint ' + __VERSION__ + '\n') + sys.stdout.write('Python ' + sys.version + '\n') + sys.exit(0) + +def PrintCategories(): + """Prints a list of all the error-categories used by error messages. + + These are the categories used to filter messages via --filter. + """ + sys.stderr.write(''.join(' %s\n' % cat for cat in _ERROR_CATEGORIES)) + sys.exit(0) + + +def ParseArguments(args): + """Parses the command line arguments. + + This may set the output format and verbosity level as side-effects. + + Args: + args: The command line arguments: + + Returns: + The list of filenames to lint. + """ + try: + (opts, filenames) = getopt.getopt(args, '', ['help', 'output=', 'verbose=', + 'v=', + 'version', + 'counting=', + 'filter=', + 'root=', + 'repository=', + 'linelength=', + 'extensions=', + 'exclude=', + 'recursive', + 'headers=', + 'quiet']) + except getopt.GetoptError: + PrintUsage('Invalid arguments.') + + verbosity = _VerboseLevel() + output_format = _OutputFormat() + filters = '' + quiet = _Quiet() + counting_style = '' + recursive = False + + for (opt, val) in opts: + if opt == '--help': + PrintUsage(None) + if opt == '--version': + PrintVersion() + elif opt == '--output': + if val not in ('emacs', 'vs7', 'eclipse', 'junit'): + PrintUsage('The only allowed output formats are emacs, vs7, eclipse ' + 'and junit.') + output_format = val + elif opt == '--quiet': + quiet = True + elif opt == '--verbose' or opt == '--v': + verbosity = int(val) + elif opt == '--filter': + filters = val + if not filters: + PrintCategories() + elif opt == '--counting': + if val not in ('total', 'toplevel', 'detailed'): + PrintUsage('Valid counting options are total, toplevel, and detailed') + counting_style = val + elif opt == '--root': + global _root + _root = val + elif opt == '--repository': + global _repository + _repository = val + elif opt == '--linelength': + global _line_length + try: + _line_length = int(val) + except ValueError: + PrintUsage('Line length must be digits.') + elif opt == '--exclude': + global _excludes + if not _excludes: + _excludes = set() + _excludes.update(glob.glob(val)) + elif opt == '--extensions': + global _valid_extensions + try: + _valid_extensions = set(val.split(',')) + except ValueError: + PrintUsage('Extensions must be comma seperated list.') + elif opt == '--headers': + ProcessHppHeadersOption(val) + elif opt == '--recursive': + recursive = True + + if not filenames: + PrintUsage('No files were specified.') + + if recursive: + filenames = _ExpandDirectories(filenames) + + if _excludes: + filenames = _FilterExcludedFiles(filenames) + + _SetOutputFormat(output_format) + _SetQuiet(quiet) + _SetVerboseLevel(verbosity) + _SetFilters(filters) + _SetCountingStyle(counting_style) + + return filenames + +def _ExpandDirectories(filenames): + """Searches a list of filenames and replaces directories in the list with + all files descending from those directories. Files with extensions not in + the valid extensions list are excluded. + + Args: + filenames: A list of files or directories + + Returns: + A list of all files that are members of filenames or descended from a + directory in filenames + """ + expanded = set() + for filename in filenames: + if not os.path.isdir(filename): + expanded.add(filename) + continue + + for root, _, files in os.walk(filename): + for loopfile in files: + fullname = os.path.join(root, loopfile) + if fullname.startswith('.' + os.path.sep): + fullname = fullname[len('.' + os.path.sep):] + expanded.add(fullname) + + filtered = [] + for filename in expanded: + if os.path.splitext(filename)[1][1:] in GetAllExtensions(): + filtered.append(filename) + + return filtered + +def _FilterExcludedFiles(filenames): + """Filters out files listed in the --exclude command line switch. File paths + in the switch are evaluated relative to the current working directory + """ + exclude_paths = [os.path.abspath(f) for f in _excludes] + return [f for f in filenames if os.path.abspath(f) not in exclude_paths] + +def main(): + filenames = ParseArguments(sys.argv[1:]) + backup_err = sys.stderr + try: + # Change stderr to write with replacement characters so we don't die + # if we try to print something containing non-ASCII characters. + sys.stderr = codecs.StreamReader(sys.stderr, 'replace') + + _cpplint_state.ResetErrorCounts() + for filename in filenames: + ProcessFile(filename, _cpplint_state.verbose_level) + # If --quiet is passed, suppress printing error count unless there are errors. + if not _cpplint_state.quiet or _cpplint_state.error_count > 0: + _cpplint_state.PrintErrorCounts() + + if _cpplint_state.output_format == 'junit': + sys.stderr.write(_cpplint_state.FormatJUnitXML()) + + finally: + sys.stderr = backup_err + + sys.exit(_cpplint_state.error_count > 0) + + +if __name__ == '__main__': + main() From d6c4cd3b6daaa0c0671f2f8b86f4ccb2a3469d5b Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 00:30:08 +0100 Subject: [PATCH 036/220] :rotating_light: adding targets for static analyzers --- Makefile | 205 +++++++++++++++++++++++++++++++++++++++++++++++++++---- 1 file changed, 192 insertions(+), 13 deletions(-) diff --git a/Makefile b/Makefile index f16b209a..7d2bfb91 100644 --- a/Makefile +++ b/Makefile @@ -49,6 +49,9 @@ all: @echo "clean - remove built files" @echo "coverage - create coverage information with lcov" @echo "cppcheck - analyze code with cppcheck" + @echo "cpplint - analyze code with cpplint" + @echo "clang_tidy - analyze code with Clang-Tidy" + @echo "clang_analyze - analyze code with Clang-Analyzer" @echo "doctest - compile example files and check their output" @echo "fuzz_testing - prepare fuzz testing of the JSON parser" @echo "fuzz_testing_bson - prepare fuzz testing of the BSON parser" @@ -123,7 +126,7 @@ doctest: # -Wno-c++2a-compat: u8 literals will behave differently in C++20... # -Wno-padded: padding is nothing to warn about pedantic_clang: - $(MAKE) json_unit CXX=$(COMPILER_DIR)/clang++ CXXFLAGS="\ + $(MAKE) json_unit CXX=$(COMPILER_DIR)/clang++ CXXFLAGS=" \ -std=c++11 -Wno-c++98-compat -Wno-c++98-compat-pedantic \ -Werror \ -Weverything \ @@ -141,68 +144,229 @@ pedantic_clang: # calling GCC with most warnings pedantic_gcc: - $(MAKE) json_unit CXX=$(COMPILER_DIR)/g++ CXXFLAGS="\ + $(MAKE) json_unit CXX=$(COMPILER_DIR)/g++ CXXFLAGS=" \ -std=c++11 \ - -Wno-deprecated-declarations \ - -Werror \ - -Wall -Wpedantic -Wextra \ + -Waddress \ + -Waddress-of-packed-member \ + -Waggressive-loop-optimizations \ + -Waligned-new=all \ + -Wall \ + -Walloc-zero \ -Walloca \ + -Warray-bounds \ -Warray-bounds=2 \ - -Wcast-qual -Wcast-align \ + -Wattribute-alias=2 \ + -Wattribute-warning \ + -Wattributes \ + -Wbool-compare \ + -Wbool-operation \ + -Wbuiltin-declaration-mismatch \ + -Wbuiltin-macro-redefined \ + -Wcannot-profile \ + -Wcast-align \ + -Wcast-function-type \ + -Wcast-qual \ + -Wcatch-value=3 \ -Wchar-subscripts \ + -Wclass-conversion \ + -Wclass-memaccess \ + -Wclobbered \ + -Wcomment \ + -Wcomments \ -Wconditionally-supported \ -Wconversion \ + -Wconversion-null \ + -Wcoverage-mismatch \ + -Wcpp \ + -Wctor-dtor-privacy \ + -Wdangling-else \ -Wdate-time \ + -Wdelete-incomplete \ + -Wdelete-non-virtual-dtor \ -Wdeprecated \ + -Wdeprecated-copy \ + -Wdeprecated-copy-dtor \ + -Wdeprecated-declarations \ -Wdisabled-optimization \ + -Wdiv-by-zero \ -Wdouble-promotion \ -Wduplicated-branches \ -Wduplicated-cond \ + -Wempty-body \ + -Wendif-labels \ + -Wenum-compare \ + -Werror \ + -Wextra \ + -Wextra-semi \ + -Wfloat-conversion \ + -Wfloat-equal \ + -Wformat \ + -Wformat-contains-nul \ + -Wformat-extra-args \ + -Wformat-nonliteral \ -Wformat-overflow=2 \ + -Wformat-security \ -Wformat-signedness \ -Wformat-truncation=2 \ + -Wformat-y2k \ + -Wformat-zero-length \ -Wformat=2 \ - -Wno-ignored-qualifiers \ + -Wframe-address \ + -Wfree-nonheap-object \ + -Whsa \ + -Wif-not-aligned \ + -Wignored-attributes \ + -Wignored-qualifiers \ -Wimplicit-fallthrough=5 \ + -Winherited-variadic-ctor \ + -Winit-list-lifetime \ + -Winit-self \ + -Winline \ + -Wint-in-bool-context \ + -Wint-to-pointer-cast \ + -Winvalid-memory-model \ + -Winvalid-offsetof \ + -Winvalid-pch \ + -Wliteral-suffix \ + -Wlogical-not-parentheses \ -Wlogical-op \ + -Wlto-type-mismatch \ + -Wmain \ + -Wmaybe-uninitialized \ + -Wmemset-elt-size \ + -Wmemset-transposed-args \ + -Wmisleading-indentation \ + -Wmissing-attributes \ + -Wmissing-braces \ -Wmissing-declarations \ + -Wmissing-field-initializers \ -Wmissing-format-attribute \ -Wmissing-include-dirs \ + -Wmissing-noreturn \ + -Wmissing-profile \ + -Wmultichar \ + -Wmultiple-inheritance \ + -Wmultistatement-macros \ + -Wnarrowing \ + -Wno-deprecated-declarations \ + -Wno-effc++ \ + -Wno-expansion-to-defined \ + -Wno-ignored-qualifiers \ + -Wno-long-long \ + -Wno-namespaces \ + -Wno-padded \ + -Wno-switch-default \ + -Wno-switch-enum \ + -Wno-system-headers \ + -Wno-templates \ + -Wno-undef \ -Wnoexcept \ + -Wnoexcept-type \ + -Wnon-template-friend \ + -Wnon-virtual-dtor \ -Wnonnull \ + -Wnonnull-compare \ + -Wnonportable-cfstrings \ + -Wnormalized \ -Wnull-dereference \ + -Wodr \ -Wold-style-cast \ + -Wopenmp-simd \ + -Woverflow \ + -Woverlength-strings \ -Woverloaded-virtual \ + -Wpacked \ + -Wpacked-bitfield-compat \ + -Wpacked-not-aligned \ -Wparentheses \ + -Wpedantic \ + -Wpessimizing-move \ -Wplacement-new=2 \ + -Wpmf-conversions \ + -Wpointer-arith \ + -Wpointer-compare \ + -Wpragmas \ + -Wprio-ctor-dtor \ + -Wpsabi \ -Wredundant-decls \ + -Wredundant-move \ + -Wregister \ -Wreorder \ -Wrestrict \ + -Wreturn-local-addr \ + -Wreturn-type \ + -Wscalar-storage-order \ + -Wsequence-point \ + -Wshadow \ + -Wshadow-compatible-local \ + -Wshadow-local \ + -Wshadow=compatible-local \ -Wshadow=global \ + -Wshadow=local \ + -Wshift-count-negative \ + -Wshift-count-overflow \ + -Wshift-negative-value \ -Wshift-overflow=2 \ + -Wsign-compare \ -Wsign-conversion \ -Wsign-promo \ -Wsized-deallocation \ + -Wsizeof-array-argument \ + -Wsizeof-pointer-div \ + -Wsizeof-pointer-memaccess \ + -Wstack-protector \ + -Wstrict-aliasing=3 \ + -Wstrict-null-sentinel \ -Wstrict-overflow=5 \ + -Wstringop-overflow=4 \ + -Wstringop-truncation \ + -Wsubobject-linkage \ + -Wsuggest-attribute=cold \ -Wsuggest-attribute=const \ -Wsuggest-attribute=format \ + -Wsuggest-attribute=malloc \ -Wsuggest-attribute=noreturn \ -Wsuggest-attribute=pure \ -Wsuggest-final-methods \ -Wsuggest-final-types \ -Wsuggest-override \ + -Wswitch \ + -Wswitch-bool \ + -Wswitch-unreachable \ + -Wsync-nand \ + -Wsynth \ + -Wtautological-compare \ + -Wterminate \ + -Wtrampolines \ -Wtrigraphs \ - -Wundef \ - -Wuninitialized -Wunknown-pragmas \ + -Wtype-limits \ + -Wuninitialized \ + -Wunknown-pragmas \ + -Wunreachable-code \ + -Wunsafe-loop-optimizations \ -Wunused \ + -Wunused-but-set-parameter \ + -Wunused-but-set-variable \ -Wunused-const-variable=2 \ + -Wunused-function \ + -Wunused-label \ + -Wunused-local-typedefs \ -Wunused-macros \ -Wunused-parameter \ + -Wunused-result \ + -Wunused-value \ + -Wunused-variable \ -Wuseless-cast \ + -Wvarargs \ -Wvariadic-macros \ - -Wctor-dtor-privacy \ - -Winit-self \ - -Wstrict-null-sentinel" + -Wvector-operation-performance \ + -Wvirtual-inheritance \ + -Wvirtual-move-assign \ + -Wvla \ + -Wvolatile-register-var \ + -Wwrite-strings \ + -Wzero-as-null-pointer-constant \ + " ########################################################################## # benchmarks @@ -279,7 +443,7 @@ fuzzing-stop: # call cppcheck on the main header file cppcheck: - cppcheck --enable=warning --inconclusive --force --std=c++11 $(AMALGAMATED_FILE) --error-exitcode=1 + cppcheck --enable=warning --inconclusive --force --std=c++11 $(SRCS) --error-exitcode=1 # compile and check with Clang Static Analyzer clang_analyze: @@ -288,6 +452,21 @@ clang_analyze: cd clang_analyze_build ; CCC_CXX=$(COMPILER_DIR)/clang++ CXX=$(COMPILER_DIR)/clang++ $(COMPILER_DIR)/scan-build cmake .. -GNinja cd clang_analyze_build ; $(COMPILER_DIR)/scan-build -enable-checker alpha.core.BoolAssignment,alpha.core.CallAndMessageUnInitRefArg,alpha.core.CastSize,alpha.core.CastToStruct,alpha.core.Conversion,alpha.core.DynamicTypeChecker,alpha.core.FixedAddr,alpha.core.PointerArithm,alpha.core.PointerSub,alpha.core.SizeofPtr,alpha.core.StackAddressAsyncEscape,alpha.core.TestAfterDivZero,alpha.deadcode.UnreachableCode,core.builtin.BuiltinFunctions,core.builtin.NoReturnFunctions,core.CallAndMessage,core.DivideZero,core.DynamicTypePropagation,core.NonnilStringConstants,core.NonNullParamChecker,core.NullDereference,core.StackAddressEscape,core.UndefinedBinaryOperatorResult,core.uninitialized.ArraySubscript,core.uninitialized.Assign,core.uninitialized.Branch,core.uninitialized.CapturedBlockVariable,core.uninitialized.UndefReturn,core.VLASize,cplusplus.InnerPointer,cplusplus.Move,cplusplus.NewDelete,cplusplus.NewDeleteLeaks,cplusplus.SelfAssignment,deadcode.DeadStores,nullability.NullableDereferenced,nullability.NullablePassedToNonnull,nullability.NullableReturnedFromNonnull,nullability.NullPassedToNonnull,nullability.NullReturnedFromNonnull --use-c++=$(COMPILER_DIR)/clang++ --view -analyze-headers -o clang_analyze_build/report.html ninja +# call cpplint (some errors expected due to false positives) +cpplint: + third_party/cpplint/cpplint.py --filter=-whitespace,-legal,-readability/alt_tokens,-runtime/references,-runtime/explicit --quiet --recursive include + +clang_tidy: + $(COMPILER_DIR)/clang-tidy -checks='-*, readability-*, -readability-magic-numbers, -readability-uppercase-literal-suffix, modernize-*, bugprone-*, performance-*, clang-analyzer-*, portability-*, cert-*, hicpp-*, -hicpp-no-array-decay, -hicpp-uppercase-literal-suffix, google-*, -google-runtime-references' $(SRCS) -- -Iinclude -std=c++11 + +pvs_studio: + rm -fr pvs_studio_build + mkdir pvs_studio_build + cd pvs_studio_build ; cmake .. -DCMAKE_EXPORT_COMPILE_COMMANDS=On + cd pvs_studio_build ; pvs-studio-analyzer analyze -j 10 + cd pvs_studio_build ; plog-converter -a'GA:1,2;64:1;CS' -t fullhtml PVS-Studio.log -o pvs + open pvs_studio_build/pvs/index.html + ########################################################################## # maintainer targets ########################################################################## From b8451c236f45aacf7cbcb29354b4021b46ae9b19 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 12:01:49 +0100 Subject: [PATCH 037/220] :rotating_light: fixed warnings --- .../nlohmann/detail/conversions/to_json.hpp | 2 +- include/nlohmann/detail/exceptions.hpp | 52 +- .../nlohmann/detail/input/binary_reader.hpp | 9 +- .../nlohmann/detail/input/input_adapters.hpp | 10 +- include/nlohmann/detail/input/json_sax.hpp | 22 +- include/nlohmann/detail/input/lexer.hpp | 6 +- include/nlohmann/detail/input/parser.hpp | 4 +- .../nlohmann/detail/iterators/iter_impl.hpp | 4 +- .../detail/iterators/iteration_proxy.hpp | 4 +- include/nlohmann/detail/json_pointer.hpp | 2 +- include/nlohmann/detail/meta/detected.hpp | 2 + include/nlohmann/detail/meta/type_traits.hpp | 4 +- include/nlohmann/detail/output/serializer.hpp | 17 +- include/nlohmann/json.hpp | 30 +- single_include/nlohmann/json.hpp | 9161 +++++++++-------- test/src/unit-alt-string.cpp | 2 +- test/src/unit-class_parser.cpp | 2 +- test/src/unit-deserialization.cpp | 2 +- test/src/unit-regression.cpp | 8 +- 19 files changed, 4721 insertions(+), 4622 deletions(-) diff --git a/include/nlohmann/detail/conversions/to_json.hpp b/include/nlohmann/detail/conversions/to_json.hpp index 96095e9c..227d7455 100644 --- a/include/nlohmann/detail/conversions/to_json.hpp +++ b/include/nlohmann/detail/conversions/to_json.hpp @@ -10,10 +10,10 @@ #include // valarray #include // vector +#include #include #include #include -#include namespace nlohmann { diff --git a/include/nlohmann/detail/exceptions.hpp b/include/nlohmann/detail/exceptions.hpp index 1ac2606b..5a73dedb 100644 --- a/include/nlohmann/detail/exceptions.hpp +++ b/include/nlohmann/detail/exceptions.hpp @@ -102,12 +102,12 @@ json.exception.parse_error.114 | parse error: Unsupported BSON record type 0x0F @liveexample{The following code shows how a `parse_error` exception can be caught.,parse_error} -@sa @ref exception for the base class of the library exceptions -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with +@sa - @ref exception for the base class of the library exceptions +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors @since version 3.0.0 */ @@ -117,7 +117,7 @@ class parse_error : public exception /*! @brief create a parse error exception @param[in] id_ the id of the exception - @param[in] position the position where the error occurred (or with + @param[in] pos the position where the error occurred (or with chars_read_total=0 if the position cannot be determined) @param[in] what_arg the explanatory string @@ -188,12 +188,12 @@ json.exception.invalid_iterator.214 | cannot get value | Cannot get value for it @liveexample{The following code shows how an `invalid_iterator` exception can be caught.,invalid_iterator} -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref type_error for exceptions indicating executing a member function with +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref type_error for exceptions indicating executing a member function with a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors @since version 3.0.0 */ @@ -242,11 +242,11 @@ json.exception.type_error.317 | JSON value cannot be serialized to requested for @liveexample{The following code shows how a `type_error` exception can be caught.,type_error} -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors @since version 3.0.0 */ @@ -287,12 +287,12 @@ json.exception.out_of_range.409 | BSON key cannot contain code point U+0000 (at @liveexample{The following code shows how an `out_of_range` exception can be caught.,out_of_range} -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with a wrong type -@sa @ref other_error for exceptions indicating other library errors +@sa - @ref other_error for exceptions indicating other library errors @since version 3.0.0 */ @@ -321,12 +321,12 @@ name / id | example message | description ------------------------------ | --------------- | ------------------------- json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref out_of_range for exceptions indicating access out of the defined range @liveexample{The following code shows how an `other_error` exception can be caught.,other_error} diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index ff6b1ab6..cded3b54 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -13,9 +13,9 @@ #include // char_traits, string #include // make_pair, move +#include #include #include -#include #include #include #include @@ -52,6 +52,13 @@ class binary_reader assert(ia); } + // make class move-only + binary_reader(const binary_reader&) = delete; + binary_reader(binary_reader&&) noexcept = default; + binary_reader& operator=(const binary_reader&) = delete; + binary_reader& operator=(binary_reader&&) noexcept = default; + ~binary_reader() = default; + /*! @param[in] format the binary format to parse @param[in] sax_ a SAX event processor diff --git a/include/nlohmann/detail/input/input_adapters.hpp b/include/nlohmann/detail/input/input_adapters.hpp index c3d33391..0374731f 100644 --- a/include/nlohmann/detail/input/input_adapters.hpp +++ b/include/nlohmann/detail/input/input_adapters.hpp @@ -3,6 +3,7 @@ #include // array #include // assert #include // size_t +#include //FILE * #include // strlen #include // istream #include // begin, end, iterator_traits, random_access_iterator_tag, distance, next @@ -11,7 +12,6 @@ #include // string, char_traits #include // enable_if, is_base_of, is_pointer, is_integral, remove_pointer #include // pair, declval -#include //FILE * #include #include @@ -59,10 +59,18 @@ class file_input_adapter : public input_adapter_protocol : m_file(f) {} + // make class move-only + file_input_adapter(const file_input_adapter&) = delete; + file_input_adapter(file_input_adapter&&) noexcept = default; + file_input_adapter& operator=(const file_input_adapter&) = delete; + file_input_adapter& operator=(file_input_adapter&&) noexcept = default; + ~file_input_adapter() override = default; + std::char_traits::int_type get_character() noexcept override { return std::fgetc(m_file); } + private: /// the file pointer to read from std::FILE* m_file; diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index d267bf2b..097b33e5 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -159,6 +159,13 @@ class json_sax_dom_parser : root(r), allow_exceptions(allow_exceptions_) {} + // make class move-only + json_sax_dom_parser(const json_sax_dom_parser&) = delete; + json_sax_dom_parser(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser& operator=(const json_sax_dom_parser&) = delete; + json_sax_dom_parser& operator=(json_sax_dom_parser&&) noexcept = default; + ~json_sax_dom_parser() = default; + bool null() { handle_value(nullptr); @@ -307,7 +314,7 @@ class json_sax_dom_parser /// the parsed JSON value BasicJsonType& root; /// stack to model hierarchy of values - std::vector ref_stack; + std::vector ref_stack {}; /// helper to hold the reference for the next object element BasicJsonType* object_element = nullptr; /// whether a syntax error occurred @@ -335,6 +342,13 @@ class json_sax_dom_callback_parser keep_stack.push_back(true); } + // make class move-only + json_sax_dom_callback_parser(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser(json_sax_dom_callback_parser&&) = default; + json_sax_dom_callback_parser& operator=(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser& operator=(json_sax_dom_callback_parser&&) = default; + ~json_sax_dom_callback_parser() = default; + bool null() { handle_value(nullptr); @@ -612,11 +626,11 @@ class json_sax_dom_callback_parser /// the parsed JSON value BasicJsonType& root; /// stack to model hierarchy of values - std::vector ref_stack; + std::vector ref_stack {}; /// stack to manage which values to keep - std::vector keep_stack; + std::vector keep_stack {}; /// stack to manage which object keys to keep - std::vector key_keep_stack; + std::vector key_keep_stack {}; /// helper to hold the reference for the next object element BasicJsonType* object_element = nullptr; /// whether a syntax error occurred diff --git a/include/nlohmann/detail/input/lexer.hpp b/include/nlohmann/detail/input/lexer.hpp index a6bde97e..d056b8f5 100644 --- a/include/nlohmann/detail/input/lexer.hpp +++ b/include/nlohmann/detail/input/lexer.hpp @@ -3,16 +3,16 @@ #include // array #include // localeconv #include // size_t -#include // strtof, strtod, strtold, strtoll, strtoull #include // snprintf +#include // strtof, strtod, strtold, strtoll, strtoull #include // initializer_list #include // char_traits, string #include // move #include // vector -#include #include #include +#include namespace nlohmann { @@ -1485,7 +1485,7 @@ scan_number_done: bool next_unget = false; /// the start position of the current token - position_t position; + position_t position {}; /// raw input token string (for error messages) std::vector token_string {}; diff --git a/include/nlohmann/detail/input/parser.hpp b/include/nlohmann/detail/input/parser.hpp index e1999871..72895c1d 100644 --- a/include/nlohmann/detail/input/parser.hpp +++ b/include/nlohmann/detail/input/parser.hpp @@ -9,11 +9,11 @@ #include // vector #include -#include -#include #include #include #include +#include +#include #include namespace nlohmann diff --git a/include/nlohmann/detail/iterators/iter_impl.hpp b/include/nlohmann/detail/iterators/iter_impl.hpp index 04d7cacb..0b0282c2 100644 --- a/include/nlohmann/detail/iterators/iter_impl.hpp +++ b/include/nlohmann/detail/iterators/iter_impl.hpp @@ -607,7 +607,7 @@ class iter_impl /// associated JSON instance pointer m_object = nullptr; /// the actual iterator of the associated instance - internal_iterator::type> m_it; + internal_iterator::type> m_it {}; }; } // namespace detail -} // namespace nlohmann \ No newline at end of file +} // namespace nlohmann diff --git a/include/nlohmann/detail/iterators/iteration_proxy.hpp b/include/nlohmann/detail/iterators/iteration_proxy.hpp index 48231968..da2e32b4 100644 --- a/include/nlohmann/detail/iterators/iteration_proxy.hpp +++ b/include/nlohmann/detail/iterators/iteration_proxy.hpp @@ -1,12 +1,12 @@ #pragma once #include // size_t -#include // string, to_string #include // input_iterator_tag +#include // string, to_string #include // tuple_size, get, tuple_element -#include #include +#include namespace nlohmann { diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 5dba7b11..a2db2357 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -7,8 +7,8 @@ #include // move #include // vector -#include #include +#include #include namespace nlohmann diff --git a/include/nlohmann/detail/meta/detected.hpp b/include/nlohmann/detail/meta/detected.hpp index c1bd5476..5b52460a 100644 --- a/include/nlohmann/detail/meta/detected.hpp +++ b/include/nlohmann/detail/meta/detected.hpp @@ -14,7 +14,9 @@ struct nonesuch nonesuch() = delete; ~nonesuch() = delete; nonesuch(nonesuch const&) = delete; + nonesuch(nonesuch const&&) = delete; void operator=(nonesuch const&) = delete; + void operator=(nonesuch&&) = delete; }; template // false_type, is_constructible, is_integral, is_same, true_type #include // declval -#include #include +#include #include #include -#include +#include namespace nlohmann { diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index 5e85355a..430eff62 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -14,8 +14,8 @@ #include // is_same #include // move -#include #include +#include #include #include #include @@ -278,6 +278,11 @@ class serializer o->write_characters("null", 4); return; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } } @@ -474,6 +479,11 @@ class serializer state = UTF8_ACCEPT; break; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } break; } @@ -534,6 +544,11 @@ class serializer } break; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } } } diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 3228f9a2..769126e5 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -48,29 +48,29 @@ SOFTWARE. #include // declval, forward, move, pair, swap #include // vector -#include -#include -#include -#include -#include -#include +#include #include #include +#include +#include #include #include #include -#include #include #include #include #include -#include -#include -#include -#include -#include +#include #include -#include +#include +#include +#include +#include +#include +#include +#include +#include +#include /*! @brief namespace for Niels Lohmann @@ -1325,6 +1325,8 @@ class basic_json case value_t::discarded: m_type = value_t::discarded; break; + default: + assert(false); // LCOV_EXCL_LINE } assert_invariant(); } @@ -7695,7 +7697,7 @@ class basic_json break; } - case patch_operations::invalid: + default: { // op must be "add", "remove", "replace", "move", "copy", or // "test" diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 897c22d0..8e677438 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -48,72 +48,406 @@ SOFTWARE. #include // declval, forward, move, pair, swap #include // vector -// #include -#ifndef INCLUDE_NLOHMANN_JSON_FWD_HPP_ -#define INCLUDE_NLOHMANN_JSON_FWD_HPP_ +// #include -#include // int64_t, uint64_t + +#include + +// #include + + +#include // transform +#include // array +#include // and, not +#include // forward_list +#include // inserter, front_inserter, end #include // map -#include // allocator #include // string -#include // vector +#include // tuple, make_tuple +#include // is_arithmetic, is_same, is_enum, underlying_type, is_convertible +#include // unordered_map +#include // pair, declval +#include // valarray + +// #include + + +#include // exception +#include // runtime_error +#include // to_string + +// #include + + +#include // size_t -/*! -@brief namespace for Niels Lohmann -@see https://github.com/nlohmann -@since version 1.0.0 -*/ namespace nlohmann { -/*! -@brief default JSONSerializer template argument +namespace detail +{ +/// struct to capture the start position of the current token +struct position_t +{ + /// the total number of characters read + std::size_t chars_read_total = 0; + /// the number of characters read in the current line + std::size_t chars_read_current_line = 0; + /// the number of lines read + std::size_t lines_read = 0; -This serializer ignores the template arguments and uses ADL -([argument-dependent lookup](https://en.cppreference.com/w/cpp/language/adl)) -for serialization. -*/ -template -struct adl_serializer; + /// conversion to size_t to preserve SAX interface + constexpr operator size_t() const + { + return chars_read_total; + } +}; -template class ObjectType = - std::map, - template class ArrayType = std::vector, - class StringType = std::string, class BooleanType = bool, - class NumberIntegerType = std::int64_t, - class NumberUnsignedType = std::uint64_t, - class NumberFloatType = double, - template class AllocatorType = std::allocator, - template class JSONSerializer = - adl_serializer> -class basic_json; +} // namespace detail +} // namespace nlohmann + + +namespace nlohmann +{ +namespace detail +{ +//////////////// +// exceptions // +//////////////// /*! -@brief JSON Pointer +@brief general exception of the @ref basic_json class -A JSON pointer defines a string syntax for identifying a specific value -within a JSON document. It can be used with functions `at` and -`operator[]`. Furthermore, JSON pointers are the base for JSON patches. +This class is an extension of `std::exception` objects with a member @a id for +exception ids. It is used as the base class for all exceptions thrown by the +@ref basic_json class. This class can hence be used as "wildcard" to catch +exceptions. -@sa [RFC 6901](https://tools.ietf.org/html/rfc6901) +Subclasses: +- @ref parse_error for exceptions indicating a parse error +- @ref invalid_iterator for exceptions indicating errors with iterators +- @ref type_error for exceptions indicating executing a member function with + a wrong type +- @ref out_of_range for exceptions indicating access out of the defined range +- @ref other_error for exceptions indicating other library errors -@since version 2.0.0 +@internal +@note To have nothrow-copy-constructible exceptions, we internally use + `std::runtime_error` which can cope with arbitrary-length error messages. + Intermediate strings are built with static functions and then passed to + the actual constructor. +@endinternal + +@liveexample{The following code shows how arbitrary library exceptions can be +caught.,exception} + +@since version 3.0.0 */ -template -class json_pointer; +class exception : public std::exception +{ + public: + /// returns the explanatory string + const char* what() const noexcept override + { + return m.what(); + } + + /// the id of the exception + const int id; + + protected: + exception(int id_, const char* what_arg) : id(id_), m(what_arg) {} + + static std::string name(const std::string& ename, int id_) + { + return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + } + + private: + /// an exception object as storage for error messages + std::runtime_error m; +}; /*! -@brief default JSON class +@brief exception indicating a parse error -This type is the default specialization of the @ref basic_json class which -uses the standard template types. +This exception is thrown by the library when a parse error occurs. Parse errors +can occur during the deserialization of JSON text, CBOR, MessagePack, as well +as when using JSON Patch. -@since version 1.0.0 +Member @a byte holds the byte index of the last read character in the input +file. + +Exceptions have ids 1xx. + +name / id | example message | description +------------------------------ | --------------- | ------------------------- +json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. +json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number without a leading `0`. +json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. +json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. +json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xF8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. +json.exception.parse_error.114 | parse error: Unsupported BSON record type 0x0F | The parsing of the corresponding BSON record type is not implemented (yet). + +@note For an input with n bytes, 1 is the index of the first character and n+1 + is the index of the terminating null byte or the end of file. This also + holds true when reading a byte vector (CBOR or MessagePack). + +@liveexample{The following code shows how a `parse_error` exception can be +caught.,parse_error} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 */ -using json = basic_json<>; +class parse_error : public exception +{ + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] pos the position where the error occurred (or with + chars_read_total=0 if the position cannot be + determined) + @param[in] what_arg the explanatory string + @return parse_error object + */ + static parse_error create(int id_, const position_t& pos, const std::string& what_arg) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + position_string(pos) + ": " + what_arg; + return parse_error(id_, pos.chars_read_total, w.c_str()); + } + + static parse_error create(int id_, std::size_t byte_, const std::string& what_arg) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + (byte_ != 0 ? (" at byte " + std::to_string(byte_)) : "") + + ": " + what_arg; + return parse_error(id_, byte_, w.c_str()); + } + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character and + n+1 is the index of the terminating null byte or the end of file. + This also holds true when reading a byte vector (CBOR or MessagePack). + */ + const std::size_t byte; + + private: + parse_error(int id_, std::size_t byte_, const char* what_arg) + : exception(id_, what_arg), byte(byte_) {} + + static std::string position_string(const position_t& pos) + { + return " at line " + std::to_string(pos.lines_read + 1) + + ", column " + std::to_string(pos.chars_read_current_line); + } +}; + +/*! +@brief exception indicating errors with iterators + +This exception is thrown if iterators passed to a library function do not match +the expected semantics. + +Exceptions have ids 2xx. + +name / id | example message | description +----------------------------------- | --------------- | ------------------------- +json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. +json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compared, because JSON objects are unordered. +json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + +@liveexample{The following code shows how an `invalid_iterator` exception can be +caught.,invalid_iterator} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class invalid_iterator : public exception +{ + public: + static invalid_iterator create(int id_, const std::string& what_arg) + { + std::string w = exception::name("invalid_iterator", id_) + what_arg; + return invalid_iterator(id_, w.c_str()); + } + + private: + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating executing a member function with a wrong type + +This exception is thrown in case of a type error; that is, a library function is +executed on a JSON value whose type does not match the expected semantics. + +Exceptions have ids 3xx. + +name / id | example message | description +----------------------------- | --------------- | ------------------------- +json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. +json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. +json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.type_error.312 | cannot use update() with string | The @ref update() member functions can only be executed for certain JSON types. +json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. +json.exception.type_error.316 | invalid UTF-8 byte at index 10: 0x7E | The @ref dump function only works with UTF-8 encoded strings; that is, if you assign a `std::string` to a JSON value, make sure it is UTF-8 encoded. | +json.exception.type_error.317 | JSON value cannot be serialized to requested format | The dynamic type of the object cannot be represented in the requested serialization format (e.g. a raw `true` or `null` JSON object cannot be serialized to BSON) | + +@liveexample{The following code shows how a `type_error` exception can be +caught.,type_error} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class type_error : public exception +{ + public: + static type_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("type_error", id_) + what_arg; + return type_error(id_, w.c_str()); + } + + private: + type_error(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating access out of the defined range + +This exception is thrown in case a library function is called on an input +parameter that exceeds the expected range, for instance in case of array +indices or nonexisting object keys. + +Exceptions have ids 4xx. + +name / id | example message | description +------------------------------- | --------------- | ------------------------- +json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. +json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. +json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. +json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. +json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. +json.exception.out_of_range.407 | number overflow serializing '9223372036854775808' | UBJSON and BSON only support integer numbers up to 9223372036854775807. | +json.exception.out_of_range.408 | excessive array size: 8658170730974374167 | The size (following `#`) of an UBJSON array or object exceeds the maximal capacity. | +json.exception.out_of_range.409 | BSON key cannot contain code point U+0000 (at byte 2) | Key identifiers to be serialized to BSON cannot contain code point U+0000, since the key is stored as zero-terminated c-string | + +@liveexample{The following code shows how an `out_of_range` exception can be +caught.,out_of_range} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class out_of_range : public exception +{ + public: + static out_of_range create(int id_, const std::string& what_arg) + { + std::string w = exception::name("out_of_range", id_) + what_arg; + return out_of_range(id_, w.c_str()); + } + + private: + out_of_range(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating other library errors + +This exception is thrown in case of errors that cannot be classified with the +other exception types. + +Exceptions have ids 5xx. + +name / id | example message | description +------------------------------ | --------------- | ------------------------- +json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range + +@liveexample{The following code shows how an `other_error` exception can be +caught.,other_error} + +@since version 3.0.0 +*/ +class other_error : public exception +{ + public: + static other_error create(int id_, const std::string& what_arg) + { + std::string w = exception::name("other_error", id_) + what_arg; + return other_error(id_, w.c_str()); + } + + private: + other_error(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; +} // namespace detail } // namespace nlohmann -#endif // INCLUDE_NLOHMANN_JSON_FWD_HPP_ - // #include @@ -340,8 +674,6 @@ constexpr T static_const::value; #include // false_type, is_constructible, is_integral, is_same, true_type #include // declval -// #include - // #include @@ -410,6 +742,8 @@ struct iterator_traits::value>> } // namespace detail } // namespace nlohmann +// #include + // #include // #include @@ -430,7 +764,9 @@ struct nonesuch nonesuch() = delete; ~nonesuch() = delete; nonesuch(nonesuch const&) = delete; + nonesuch(nonesuch const&&) = delete; void operator=(nonesuch const&) = delete; + void operator=(nonesuch&&) = delete; }; template +// #include +#ifndef INCLUDE_NLOHMANN_JSON_FWD_HPP_ +#define INCLUDE_NLOHMANN_JSON_FWD_HPP_ + +#include // int64_t, uint64_t +#include // map +#include // allocator +#include // string +#include // vector + +/*! +@brief namespace for Niels Lohmann +@see https://github.com/nlohmann +@since version 1.0.0 +*/ +namespace nlohmann +{ +/*! +@brief default JSONSerializer template argument + +This serializer ignores the template arguments and uses ADL +([argument-dependent lookup](https://en.cppreference.com/w/cpp/language/adl)) +for serialization. +*/ +template +struct adl_serializer; + +template class ObjectType = + std::map, + template class ArrayType = std::vector, + class StringType = std::string, class BooleanType = bool, + class NumberIntegerType = std::int64_t, + class NumberUnsignedType = std::uint64_t, + class NumberFloatType = double, + template class AllocatorType = std::allocator, + template class JSONSerializer = + adl_serializer> +class basic_json; + +/*! +@brief JSON Pointer + +A JSON pointer defines a string syntax for identifying a specific value +within a JSON document. It can be used with functions `at` and +`operator[]`. Furthermore, JSON pointers are the base for JSON patches. + +@sa [RFC 6901](https://tools.ietf.org/html/rfc6901) + +@since version 2.0.0 +*/ +template +class json_pointer; + +/*! +@brief default JSON class + +This type is the default specialization of the @ref basic_json class which +uses the standard template types. + +@since version 1.0.0 +*/ +using json = basic_json<>; +} // namespace nlohmann + +#endif // INCLUDE_NLOHMANN_JSON_FWD_HPP_ namespace nlohmann @@ -810,385 +1210,6 @@ struct is_compatible_type } // namespace detail } // namespace nlohmann -// #include - - -#include // exception -#include // runtime_error -#include // to_string - -// #include - - -#include // size_t - -namespace nlohmann -{ -namespace detail -{ -/// struct to capture the start position of the current token -struct position_t -{ - /// the total number of characters read - std::size_t chars_read_total = 0; - /// the number of characters read in the current line - std::size_t chars_read_current_line = 0; - /// the number of lines read - std::size_t lines_read = 0; - - /// conversion to size_t to preserve SAX interface - constexpr operator size_t() const - { - return chars_read_total; - } -}; - -} // namespace detail -} // namespace nlohmann - - -namespace nlohmann -{ -namespace detail -{ -//////////////// -// exceptions // -//////////////// - -/*! -@brief general exception of the @ref basic_json class - -This class is an extension of `std::exception` objects with a member @a id for -exception ids. It is used as the base class for all exceptions thrown by the -@ref basic_json class. This class can hence be used as "wildcard" to catch -exceptions. - -Subclasses: -- @ref parse_error for exceptions indicating a parse error -- @ref invalid_iterator for exceptions indicating errors with iterators -- @ref type_error for exceptions indicating executing a member function with - a wrong type -- @ref out_of_range for exceptions indicating access out of the defined range -- @ref other_error for exceptions indicating other library errors - -@internal -@note To have nothrow-copy-constructible exceptions, we internally use - `std::runtime_error` which can cope with arbitrary-length error messages. - Intermediate strings are built with static functions and then passed to - the actual constructor. -@endinternal - -@liveexample{The following code shows how arbitrary library exceptions can be -caught.,exception} - -@since version 3.0.0 -*/ -class exception : public std::exception -{ - public: - /// returns the explanatory string - const char* what() const noexcept override - { - return m.what(); - } - - /// the id of the exception - const int id; - - protected: - exception(int id_, const char* what_arg) : id(id_), m(what_arg) {} - - static std::string name(const std::string& ename, int id_) - { - return "[json.exception." + ename + "." + std::to_string(id_) + "] "; - } - - private: - /// an exception object as storage for error messages - std::runtime_error m; -}; - -/*! -@brief exception indicating a parse error - -This exception is thrown by the library when a parse error occurs. Parse errors -can occur during the deserialization of JSON text, CBOR, MessagePack, as well -as when using JSON Patch. - -Member @a byte holds the byte index of the last read character in the input -file. - -Exceptions have ids 1xx. - -name / id | example message | description ------------------------------- | --------------- | ------------------------- -json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. -json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. -json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. -json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. -json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. -json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number without a leading `0`. -json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. -json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. -json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. -json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. -json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xF8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. -json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. -json.exception.parse_error.114 | parse error: Unsupported BSON record type 0x0F | The parsing of the corresponding BSON record type is not implemented (yet). - -@note For an input with n bytes, 1 is the index of the first character and n+1 - is the index of the terminating null byte or the end of file. This also - holds true when reading a byte vector (CBOR or MessagePack). - -@liveexample{The following code shows how a `parse_error` exception can be -caught.,parse_error} - -@sa @ref exception for the base class of the library exceptions -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with - a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors - -@since version 3.0.0 -*/ -class parse_error : public exception -{ - public: - /*! - @brief create a parse error exception - @param[in] id_ the id of the exception - @param[in] position the position where the error occurred (or with - chars_read_total=0 if the position cannot be - determined) - @param[in] what_arg the explanatory string - @return parse_error object - */ - static parse_error create(int id_, const position_t& pos, const std::string& what_arg) - { - std::string w = exception::name("parse_error", id_) + "parse error" + - position_string(pos) + ": " + what_arg; - return parse_error(id_, pos.chars_read_total, w.c_str()); - } - - static parse_error create(int id_, std::size_t byte_, const std::string& what_arg) - { - std::string w = exception::name("parse_error", id_) + "parse error" + - (byte_ != 0 ? (" at byte " + std::to_string(byte_)) : "") + - ": " + what_arg; - return parse_error(id_, byte_, w.c_str()); - } - - /*! - @brief byte index of the parse error - - The byte index of the last read character in the input file. - - @note For an input with n bytes, 1 is the index of the first character and - n+1 is the index of the terminating null byte or the end of file. - This also holds true when reading a byte vector (CBOR or MessagePack). - */ - const std::size_t byte; - - private: - parse_error(int id_, std::size_t byte_, const char* what_arg) - : exception(id_, what_arg), byte(byte_) {} - - static std::string position_string(const position_t& pos) - { - return " at line " + std::to_string(pos.lines_read + 1) + - ", column " + std::to_string(pos.chars_read_current_line); - } -}; - -/*! -@brief exception indicating errors with iterators - -This exception is thrown if iterators passed to a library function do not match -the expected semantics. - -Exceptions have ids 2xx. - -name / id | example message | description ------------------------------------ | --------------- | ------------------------- -json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. -json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. -json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. -json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. -json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. -json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. -json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. -json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. -json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. -json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. -json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compared, because JSON objects are unordered. -json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). - -@liveexample{The following code shows how an `invalid_iterator` exception can be -caught.,invalid_iterator} - -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref type_error for exceptions indicating executing a member function with - a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors - -@since version 3.0.0 -*/ -class invalid_iterator : public exception -{ - public: - static invalid_iterator create(int id_, const std::string& what_arg) - { - std::string w = exception::name("invalid_iterator", id_) + what_arg; - return invalid_iterator(id_, w.c_str()); - } - - private: - invalid_iterator(int id_, const char* what_arg) - : exception(id_, what_arg) {} -}; - -/*! -@brief exception indicating executing a member function with a wrong type - -This exception is thrown in case of a type error; that is, a library function is -executed on a JSON value whose type does not match the expected semantics. - -Exceptions have ids 3xx. - -name / id | example message | description ------------------------------ | --------------- | ------------------------- -json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. -json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. -json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. -json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. -json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. -json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. -json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. -json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. -json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. -json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. -json.exception.type_error.312 | cannot use update() with string | The @ref update() member functions can only be executed for certain JSON types. -json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. -json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. -json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. -json.exception.type_error.316 | invalid UTF-8 byte at index 10: 0x7E | The @ref dump function only works with UTF-8 encoded strings; that is, if you assign a `std::string` to a JSON value, make sure it is UTF-8 encoded. | -json.exception.type_error.317 | JSON value cannot be serialized to requested format | The dynamic type of the object cannot be represented in the requested serialization format (e.g. a raw `true` or `null` JSON object cannot be serialized to BSON) | - -@liveexample{The following code shows how a `type_error` exception can be -caught.,type_error} - -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref out_of_range for exceptions indicating access out of the defined range -@sa @ref other_error for exceptions indicating other library errors - -@since version 3.0.0 -*/ -class type_error : public exception -{ - public: - static type_error create(int id_, const std::string& what_arg) - { - std::string w = exception::name("type_error", id_) + what_arg; - return type_error(id_, w.c_str()); - } - - private: - type_error(int id_, const char* what_arg) : exception(id_, what_arg) {} -}; - -/*! -@brief exception indicating access out of the defined range - -This exception is thrown in case a library function is called on an input -parameter that exceeds the expected range, for instance in case of array -indices or nonexisting object keys. - -Exceptions have ids 4xx. - -name / id | example message | description -------------------------------- | --------------- | ------------------------- -json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. -json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. -json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. -json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. -json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. -json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. -json.exception.out_of_range.407 | number overflow serializing '9223372036854775808' | UBJSON and BSON only support integer numbers up to 9223372036854775807. | -json.exception.out_of_range.408 | excessive array size: 8658170730974374167 | The size (following `#`) of an UBJSON array or object exceeds the maximal capacity. | -json.exception.out_of_range.409 | BSON key cannot contain code point U+0000 (at byte 2) | Key identifiers to be serialized to BSON cannot contain code point U+0000, since the key is stored as zero-terminated c-string | - -@liveexample{The following code shows how an `out_of_range` exception can be -caught.,out_of_range} - -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with - a wrong type -@sa @ref other_error for exceptions indicating other library errors - -@since version 3.0.0 -*/ -class out_of_range : public exception -{ - public: - static out_of_range create(int id_, const std::string& what_arg) - { - std::string w = exception::name("out_of_range", id_) + what_arg; - return out_of_range(id_, w.c_str()); - } - - private: - out_of_range(int id_, const char* what_arg) : exception(id_, what_arg) {} -}; - -/*! -@brief exception indicating other library errors - -This exception is thrown in case of errors that cannot be classified with the -other exception types. - -Exceptions have ids 5xx. - -name / id | example message | description ------------------------------- | --------------- | ------------------------- -json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. - -@sa @ref exception for the base class of the library exceptions -@sa @ref parse_error for exceptions indicating a parse error -@sa @ref invalid_iterator for exceptions indicating errors with iterators -@sa @ref type_error for exceptions indicating executing a member function with - a wrong type -@sa @ref out_of_range for exceptions indicating access out of the defined range - -@liveexample{The following code shows how an `other_error` exception can be -caught.,other_error} - -@since version 3.0.0 -*/ -class other_error : public exception -{ - public: - static other_error create(int id_, const std::string& what_arg) - { - std::string w = exception::name("other_error", id_) + what_arg; - return other_error(id_, w.c_str()); - } - - private: - other_error(int id_, const char* what_arg) : exception(id_, what_arg) {} -}; -} // namespace detail -} // namespace nlohmann - // #include @@ -1268,32 +1289,6 @@ inline bool operator<(const value_t lhs, const value_t rhs) noexcept } // namespace detail } // namespace nlohmann -// #include - - -#include // transform -#include // array -#include // and, not -#include // forward_list -#include // inserter, front_inserter, end -#include // map -#include // string -#include // tuple, make_tuple -#include // is_arithmetic, is_same, is_enum, underlying_type, is_convertible -#include // unordered_map -#include // pair, declval -#include // valarray - -// #include - -// #include - -// #include - -// #include - -// #include - namespace nlohmann { @@ -1658,24 +1653,18 @@ constexpr const auto& from_json = detail::static_const::va #include // valarray #include // vector -// #include - -// #include - -// #include - // #include #include // size_t -#include // string, to_string #include // input_iterator_tag +#include // string, to_string #include // tuple_size, get, tuple_element -// #include - // #include +// #include + namespace nlohmann { @@ -1838,6 +1827,12 @@ class tuple_element> #endif } // namespace std +// #include + +// #include + +// #include + namespace nlohmann { @@ -2167,12 +2162,81 @@ constexpr const auto& to_json = detail::static_const::value; } // namespace } // namespace nlohmann + +namespace nlohmann +{ + +template +struct adl_serializer +{ + /*! + @brief convert a JSON value to any value type + + This function is usually called by the `get()` function of the + @ref basic_json class (either explicit or via conversion operators). + + @param[in] j JSON value to read from + @param[in,out] val value to write to + */ + template + static auto from_json(BasicJsonType&& j, ValueType& val) noexcept( + noexcept(::nlohmann::from_json(std::forward(j), val))) + -> decltype(::nlohmann::from_json(std::forward(j), val), void()) + { + ::nlohmann::from_json(std::forward(j), val); + } + + /*! + @brief convert any value type to a JSON value + + This function is usually called by the constructors of the @ref basic_json + class. + + @param[in,out] j JSON value to write to + @param[in] val value to read from + */ + template + static auto to_json(BasicJsonType& j, ValueType&& val) noexcept( + noexcept(::nlohmann::to_json(j, std::forward(val)))) + -> decltype(::nlohmann::to_json(j, std::forward(val)), void()) + { + ::nlohmann::to_json(j, std::forward(val)); + } +}; + +} // namespace nlohmann + +// #include + +// #include + +// #include + +// #include + + +#include // generate_n +#include // array +#include // assert +#include // ldexp +#include // size_t +#include // uint8_t, uint16_t, uint32_t, uint64_t +#include // snprintf +#include // memcpy +#include // back_inserter +#include // numeric_limits +#include // char_traits, string +#include // make_pair, move + +// #include + // #include #include // array #include // assert #include // size_t +#include //FILE * #include // strlen #include // istream #include // begin, end, iterator_traits, random_access_iterator_tag, distance, next @@ -2181,7 +2245,6 @@ constexpr const auto& to_json = detail::static_const::value; #include // string, char_traits #include // enable_if, is_base_of, is_pointer, is_integral, remove_pointer #include // pair, declval -#include //FILE * // #include @@ -2231,10 +2294,18 @@ class file_input_adapter : public input_adapter_protocol : m_file(f) {} + // make class move-only + file_input_adapter(const file_input_adapter&) = delete; + file_input_adapter(file_input_adapter&&) noexcept = default; + file_input_adapter& operator=(const file_input_adapter&) = delete; + file_input_adapter& operator=(file_input_adapter&&) noexcept = default; + ~file_input_adapter() override = default; + std::char_traits::int_type get_character() noexcept override { return std::fgetc(m_file); } + private: /// the file pointer to read from std::FILE* m_file; @@ -2602,1682 +2673,6 @@ class input_adapter } // namespace detail } // namespace nlohmann -// #include - - -#include // array -#include // localeconv -#include // size_t -#include // strtof, strtod, strtold, strtoll, strtoull -#include // snprintf -#include // initializer_list -#include // char_traits, string -#include // move -#include // vector - -// #include - -// #include - -// #include - - -namespace nlohmann -{ -namespace detail -{ -/////////// -// lexer // -/////////// - -/*! -@brief lexical analysis - -This class organizes the lexical analysis during JSON deserialization. -*/ -template -class lexer -{ - using number_integer_t = typename BasicJsonType::number_integer_t; - using number_unsigned_t = typename BasicJsonType::number_unsigned_t; - using number_float_t = typename BasicJsonType::number_float_t; - using string_t = typename BasicJsonType::string_t; - - public: - /// token types for the parser - enum class token_type - { - uninitialized, ///< indicating the scanner is uninitialized - literal_true, ///< the `true` literal - literal_false, ///< the `false` literal - literal_null, ///< the `null` literal - value_string, ///< a string -- use get_string() for actual value - value_unsigned, ///< an unsigned integer -- use get_number_unsigned() for actual value - value_integer, ///< a signed integer -- use get_number_integer() for actual value - value_float, ///< an floating point number -- use get_number_float() for actual value - begin_array, ///< the character for array begin `[` - begin_object, ///< the character for object begin `{` - end_array, ///< the character for array end `]` - end_object, ///< the character for object end `}` - name_separator, ///< the name separator `:` - value_separator, ///< the value separator `,` - parse_error, ///< indicating a parse error - end_of_input, ///< indicating the end of the input buffer - literal_or_value ///< a literal or the begin of a value (only for diagnostics) - }; - - /// return name of values of type token_type (only used for errors) - static const char* token_type_name(const token_type t) noexcept - { - switch (t) - { - case token_type::uninitialized: - return ""; - case token_type::literal_true: - return "true literal"; - case token_type::literal_false: - return "false literal"; - case token_type::literal_null: - return "null literal"; - case token_type::value_string: - return "string literal"; - case lexer::token_type::value_unsigned: - case lexer::token_type::value_integer: - case lexer::token_type::value_float: - return "number literal"; - case token_type::begin_array: - return "'['"; - case token_type::begin_object: - return "'{'"; - case token_type::end_array: - return "']'"; - case token_type::end_object: - return "'}'"; - case token_type::name_separator: - return "':'"; - case token_type::value_separator: - return "','"; - case token_type::parse_error: - return ""; - case token_type::end_of_input: - return "end of input"; - case token_type::literal_or_value: - return "'[', '{', or a literal"; - // LCOV_EXCL_START - default: // catch non-enum values - return "unknown token"; - // LCOV_EXCL_STOP - } - } - - explicit lexer(detail::input_adapter_t&& adapter) - : ia(std::move(adapter)), decimal_point_char(get_decimal_point()) {} - - // delete because of pointer members - lexer(const lexer&) = delete; - lexer(lexer&&) = delete; - lexer& operator=(lexer&) = delete; - lexer& operator=(lexer&&) = delete; - ~lexer() = default; - - private: - ///////////////////// - // locales - ///////////////////// - - /// return the locale-dependent decimal point - static char get_decimal_point() noexcept - { - const auto loc = localeconv(); - assert(loc != nullptr); - return (loc->decimal_point == nullptr) ? '.' : *(loc->decimal_point); - } - - ///////////////////// - // scan functions - ///////////////////// - - /*! - @brief get codepoint from 4 hex characters following `\u` - - For input "\u c1 c2 c3 c4" the codepoint is: - (c1 * 0x1000) + (c2 * 0x0100) + (c3 * 0x0010) + c4 - = (c1 << 12) + (c2 << 8) + (c3 << 4) + (c4 << 0) - - Furthermore, the possible characters '0'..'9', 'A'..'F', and 'a'..'f' - must be converted to the integers 0x0..0x9, 0xA..0xF, 0xA..0xF, resp. The - conversion is done by subtracting the offset (0x30, 0x37, and 0x57) - between the ASCII value of the character and the desired integer value. - - @return codepoint (0x0000..0xFFFF) or -1 in case of an error (e.g. EOF or - non-hex character) - */ - int get_codepoint() - { - // this function only makes sense after reading `\u` - assert(current == 'u'); - int codepoint = 0; - - const auto factors = { 12u, 8u, 4u, 0u }; - for (const auto factor : factors) - { - get(); - - if (current >= '0' and current <= '9') - { - codepoint += static_cast((static_cast(current) - 0x30u) << factor); - } - else if (current >= 'A' and current <= 'F') - { - codepoint += static_cast((static_cast(current) - 0x37u) << factor); - } - else if (current >= 'a' and current <= 'f') - { - codepoint += static_cast((static_cast(current) - 0x57u) << factor); - } - else - { - return -1; - } - } - - assert(0x0000 <= codepoint and codepoint <= 0xFFFF); - return codepoint; - } - - /*! - @brief check if the next byte(s) are inside a given range - - Adds the current byte and, for each passed range, reads a new byte and - checks if it is inside the range. If a violation was detected, set up an - error message and return false. Otherwise, return true. - - @param[in] ranges list of integers; interpreted as list of pairs of - inclusive lower and upper bound, respectively - - @pre The passed list @a ranges must have 2, 4, or 6 elements; that is, - 1, 2, or 3 pairs. This precondition is enforced by an assertion. - - @return true if and only if no range violation was detected - */ - bool next_byte_in_range(std::initializer_list ranges) - { - assert(ranges.size() == 2 or ranges.size() == 4 or ranges.size() == 6); - add(current); - - for (auto range = ranges.begin(); range != ranges.end(); ++range) - { - get(); - if (JSON_LIKELY(*range <= current and current <= *(++range))) - { - add(current); - } - else - { - error_message = "invalid string: ill-formed UTF-8 byte"; - return false; - } - } - - return true; - } - - /*! - @brief scan a string literal - - This function scans a string according to Sect. 7 of RFC 7159. While - scanning, bytes are escaped and copied into buffer token_buffer. Then the - function returns successfully, token_buffer is *not* null-terminated (as it - may contain \0 bytes), and token_buffer.size() is the number of bytes in the - string. - - @return token_type::value_string if string could be successfully scanned, - token_type::parse_error otherwise - - @note In case of errors, variable error_message contains a textual - description. - */ - token_type scan_string() - { - // reset token_buffer (ignore opening quote) - reset(); - - // we entered the function by reading an open quote - assert(current == '\"'); - - while (true) - { - // get next character - switch (get()) - { - // end of file while parsing string - case std::char_traits::eof(): - { - error_message = "invalid string: missing closing quote"; - return token_type::parse_error; - } - - // closing quote - case '\"': - { - return token_type::value_string; - } - - // escapes - case '\\': - { - switch (get()) - { - // quotation mark - case '\"': - add('\"'); - break; - // reverse solidus - case '\\': - add('\\'); - break; - // solidus - case '/': - add('/'); - break; - // backspace - case 'b': - add('\b'); - break; - // form feed - case 'f': - add('\f'); - break; - // line feed - case 'n': - add('\n'); - break; - // carriage return - case 'r': - add('\r'); - break; - // tab - case 't': - add('\t'); - break; - - // unicode escapes - case 'u': - { - const int codepoint1 = get_codepoint(); - int codepoint = codepoint1; // start with codepoint1 - - if (JSON_UNLIKELY(codepoint1 == -1)) - { - error_message = "invalid string: '\\u' must be followed by 4 hex digits"; - return token_type::parse_error; - } - - // check if code point is a high surrogate - if (0xD800 <= codepoint1 and codepoint1 <= 0xDBFF) - { - // expect next \uxxxx entry - if (JSON_LIKELY(get() == '\\' and get() == 'u')) - { - const int codepoint2 = get_codepoint(); - - if (JSON_UNLIKELY(codepoint2 == -1)) - { - error_message = "invalid string: '\\u' must be followed by 4 hex digits"; - return token_type::parse_error; - } - - // check if codepoint2 is a low surrogate - if (JSON_LIKELY(0xDC00 <= codepoint2 and codepoint2 <= 0xDFFF)) - { - // overwrite codepoint - codepoint = static_cast( - // high surrogate occupies the most significant 22 bits - (static_cast(codepoint1) << 10u) - // low surrogate occupies the least significant 15 bits - + static_cast(codepoint2) - // there is still the 0xD800, 0xDC00 and 0x10000 noise - // in the result so we have to subtract with: - // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 - - 0x35FDC00u); - } - else - { - error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; - return token_type::parse_error; - } - } - else - { - error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; - return token_type::parse_error; - } - } - else - { - if (JSON_UNLIKELY(0xDC00 <= codepoint1 and codepoint1 <= 0xDFFF)) - { - error_message = "invalid string: surrogate U+DC00..U+DFFF must follow U+D800..U+DBFF"; - return token_type::parse_error; - } - } - - // result of the above calculation yields a proper codepoint - assert(0x00 <= codepoint and codepoint <= 0x10FFFF); - - // translate codepoint into bytes - if (codepoint < 0x80) - { - // 1-byte characters: 0xxxxxxx (ASCII) - add(codepoint); - } - else if (codepoint <= 0x7FF) - { - // 2-byte characters: 110xxxxx 10xxxxxx - add(static_cast(0xC0u | (static_cast(codepoint) >> 6u))); - add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); - } - else if (codepoint <= 0xFFFF) - { - // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx - add(static_cast(0xE0u | (static_cast(codepoint) >> 12u))); - add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); - add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); - } - else - { - // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx - add(static_cast(0xF0u | (static_cast(codepoint) >> 18u))); - add(static_cast(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu))); - add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); - add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); - } - - break; - } - - // other characters after escape - default: - error_message = "invalid string: forbidden character after backslash"; - return token_type::parse_error; - } - - break; - } - - // invalid control characters - case 0x00: - { - error_message = "invalid string: control character U+0000 (NUL) must be escaped to \\u0000"; - return token_type::parse_error; - } - - case 0x01: - { - error_message = "invalid string: control character U+0001 (SOH) must be escaped to \\u0001"; - return token_type::parse_error; - } - - case 0x02: - { - error_message = "invalid string: control character U+0002 (STX) must be escaped to \\u0002"; - return token_type::parse_error; - } - - case 0x03: - { - error_message = "invalid string: control character U+0003 (ETX) must be escaped to \\u0003"; - return token_type::parse_error; - } - - case 0x04: - { - error_message = "invalid string: control character U+0004 (EOT) must be escaped to \\u0004"; - return token_type::parse_error; - } - - case 0x05: - { - error_message = "invalid string: control character U+0005 (ENQ) must be escaped to \\u0005"; - return token_type::parse_error; - } - - case 0x06: - { - error_message = "invalid string: control character U+0006 (ACK) must be escaped to \\u0006"; - return token_type::parse_error; - } - - case 0x07: - { - error_message = "invalid string: control character U+0007 (BEL) must be escaped to \\u0007"; - return token_type::parse_error; - } - - case 0x08: - { - error_message = "invalid string: control character U+0008 (BS) must be escaped to \\u0008 or \\b"; - return token_type::parse_error; - } - - case 0x09: - { - error_message = "invalid string: control character U+0009 (HT) must be escaped to \\u0009 or \\t"; - return token_type::parse_error; - } - - case 0x0A: - { - error_message = "invalid string: control character U+000A (LF) must be escaped to \\u000A or \\n"; - return token_type::parse_error; - } - - case 0x0B: - { - error_message = "invalid string: control character U+000B (VT) must be escaped to \\u000B"; - return token_type::parse_error; - } - - case 0x0C: - { - error_message = "invalid string: control character U+000C (FF) must be escaped to \\u000C or \\f"; - return token_type::parse_error; - } - - case 0x0D: - { - error_message = "invalid string: control character U+000D (CR) must be escaped to \\u000D or \\r"; - return token_type::parse_error; - } - - case 0x0E: - { - error_message = "invalid string: control character U+000E (SO) must be escaped to \\u000E"; - return token_type::parse_error; - } - - case 0x0F: - { - error_message = "invalid string: control character U+000F (SI) must be escaped to \\u000F"; - return token_type::parse_error; - } - - case 0x10: - { - error_message = "invalid string: control character U+0010 (DLE) must be escaped to \\u0010"; - return token_type::parse_error; - } - - case 0x11: - { - error_message = "invalid string: control character U+0011 (DC1) must be escaped to \\u0011"; - return token_type::parse_error; - } - - case 0x12: - { - error_message = "invalid string: control character U+0012 (DC2) must be escaped to \\u0012"; - return token_type::parse_error; - } - - case 0x13: - { - error_message = "invalid string: control character U+0013 (DC3) must be escaped to \\u0013"; - return token_type::parse_error; - } - - case 0x14: - { - error_message = "invalid string: control character U+0014 (DC4) must be escaped to \\u0014"; - return token_type::parse_error; - } - - case 0x15: - { - error_message = "invalid string: control character U+0015 (NAK) must be escaped to \\u0015"; - return token_type::parse_error; - } - - case 0x16: - { - error_message = "invalid string: control character U+0016 (SYN) must be escaped to \\u0016"; - return token_type::parse_error; - } - - case 0x17: - { - error_message = "invalid string: control character U+0017 (ETB) must be escaped to \\u0017"; - return token_type::parse_error; - } - - case 0x18: - { - error_message = "invalid string: control character U+0018 (CAN) must be escaped to \\u0018"; - return token_type::parse_error; - } - - case 0x19: - { - error_message = "invalid string: control character U+0019 (EM) must be escaped to \\u0019"; - return token_type::parse_error; - } - - case 0x1A: - { - error_message = "invalid string: control character U+001A (SUB) must be escaped to \\u001A"; - return token_type::parse_error; - } - - case 0x1B: - { - error_message = "invalid string: control character U+001B (ESC) must be escaped to \\u001B"; - return token_type::parse_error; - } - - case 0x1C: - { - error_message = "invalid string: control character U+001C (FS) must be escaped to \\u001C"; - return token_type::parse_error; - } - - case 0x1D: - { - error_message = "invalid string: control character U+001D (GS) must be escaped to \\u001D"; - return token_type::parse_error; - } - - case 0x1E: - { - error_message = "invalid string: control character U+001E (RS) must be escaped to \\u001E"; - return token_type::parse_error; - } - - case 0x1F: - { - error_message = "invalid string: control character U+001F (US) must be escaped to \\u001F"; - return token_type::parse_error; - } - - // U+0020..U+007F (except U+0022 (quote) and U+005C (backspace)) - case 0x20: - case 0x21: - case 0x23: - case 0x24: - case 0x25: - case 0x26: - case 0x27: - case 0x28: - case 0x29: - case 0x2A: - case 0x2B: - case 0x2C: - case 0x2D: - case 0x2E: - case 0x2F: - case 0x30: - case 0x31: - case 0x32: - case 0x33: - case 0x34: - case 0x35: - case 0x36: - case 0x37: - case 0x38: - case 0x39: - case 0x3A: - case 0x3B: - case 0x3C: - case 0x3D: - case 0x3E: - case 0x3F: - case 0x40: - case 0x41: - case 0x42: - case 0x43: - case 0x44: - case 0x45: - case 0x46: - case 0x47: - case 0x48: - case 0x49: - case 0x4A: - case 0x4B: - case 0x4C: - case 0x4D: - case 0x4E: - case 0x4F: - case 0x50: - case 0x51: - case 0x52: - case 0x53: - case 0x54: - case 0x55: - case 0x56: - case 0x57: - case 0x58: - case 0x59: - case 0x5A: - case 0x5B: - case 0x5D: - case 0x5E: - case 0x5F: - case 0x60: - case 0x61: - case 0x62: - case 0x63: - case 0x64: - case 0x65: - case 0x66: - case 0x67: - case 0x68: - case 0x69: - case 0x6A: - case 0x6B: - case 0x6C: - case 0x6D: - case 0x6E: - case 0x6F: - case 0x70: - case 0x71: - case 0x72: - case 0x73: - case 0x74: - case 0x75: - case 0x76: - case 0x77: - case 0x78: - case 0x79: - case 0x7A: - case 0x7B: - case 0x7C: - case 0x7D: - case 0x7E: - case 0x7F: - { - add(current); - break; - } - - // U+0080..U+07FF: bytes C2..DF 80..BF - case 0xC2: - case 0xC3: - case 0xC4: - case 0xC5: - case 0xC6: - case 0xC7: - case 0xC8: - case 0xC9: - case 0xCA: - case 0xCB: - case 0xCC: - case 0xCD: - case 0xCE: - case 0xCF: - case 0xD0: - case 0xD1: - case 0xD2: - case 0xD3: - case 0xD4: - case 0xD5: - case 0xD6: - case 0xD7: - case 0xD8: - case 0xD9: - case 0xDA: - case 0xDB: - case 0xDC: - case 0xDD: - case 0xDE: - case 0xDF: - { - if (JSON_UNLIKELY(not next_byte_in_range({0x80, 0xBF}))) - { - return token_type::parse_error; - } - break; - } - - // U+0800..U+0FFF: bytes E0 A0..BF 80..BF - case 0xE0: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0xA0, 0xBF, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // U+1000..U+CFFF: bytes E1..EC 80..BF 80..BF - // U+E000..U+FFFF: bytes EE..EF 80..BF 80..BF - case 0xE1: - case 0xE2: - case 0xE3: - case 0xE4: - case 0xE5: - case 0xE6: - case 0xE7: - case 0xE8: - case 0xE9: - case 0xEA: - case 0xEB: - case 0xEC: - case 0xEE: - case 0xEF: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0xBF, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // U+D000..U+D7FF: bytes ED 80..9F 80..BF - case 0xED: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0x9F, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // U+10000..U+3FFFF F0 90..BF 80..BF 80..BF - case 0xF0: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0x90, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // U+40000..U+FFFFF F1..F3 80..BF 80..BF 80..BF - case 0xF1: - case 0xF2: - case 0xF3: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // U+100000..U+10FFFF F4 80..8F 80..BF 80..BF - case 0xF4: - { - if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0x8F, 0x80, 0xBF, 0x80, 0xBF})))) - { - return token_type::parse_error; - } - break; - } - - // remaining bytes (80..C1 and F5..FF) are ill-formed - default: - { - error_message = "invalid string: ill-formed UTF-8 byte"; - return token_type::parse_error; - } - } - } - } - - static void strtof(float& f, const char* str, char** endptr) noexcept - { - f = std::strtof(str, endptr); - } - - static void strtof(double& f, const char* str, char** endptr) noexcept - { - f = std::strtod(str, endptr); - } - - static void strtof(long double& f, const char* str, char** endptr) noexcept - { - f = std::strtold(str, endptr); - } - - /*! - @brief scan a number literal - - This function scans a string according to Sect. 6 of RFC 7159. - - The function is realized with a deterministic finite state machine derived - from the grammar described in RFC 7159. Starting in state "init", the - input is read and used to determined the next state. Only state "done" - accepts the number. State "error" is a trap state to model errors. In the - table below, "anything" means any character but the ones listed before. - - state | 0 | 1-9 | e E | + | - | . | anything - ---------|----------|----------|----------|---------|---------|----------|----------- - init | zero | any1 | [error] | [error] | minus | [error] | [error] - minus | zero | any1 | [error] | [error] | [error] | [error] | [error] - zero | done | done | exponent | done | done | decimal1 | done - any1 | any1 | any1 | exponent | done | done | decimal1 | done - decimal1 | decimal2 | [error] | [error] | [error] | [error] | [error] | [error] - decimal2 | decimal2 | decimal2 | exponent | done | done | done | done - exponent | any2 | any2 | [error] | sign | sign | [error] | [error] - sign | any2 | any2 | [error] | [error] | [error] | [error] | [error] - any2 | any2 | any2 | done | done | done | done | done - - The state machine is realized with one label per state (prefixed with - "scan_number_") and `goto` statements between them. The state machine - contains cycles, but any cycle can be left when EOF is read. Therefore, - the function is guaranteed to terminate. - - During scanning, the read bytes are stored in token_buffer. This string is - then converted to a signed integer, an unsigned integer, or a - floating-point number. - - @return token_type::value_unsigned, token_type::value_integer, or - token_type::value_float if number could be successfully scanned, - token_type::parse_error otherwise - - @note The scanner is independent of the current locale. Internally, the - locale's decimal point is used instead of `.` to work with the - locale-dependent converters. - */ - token_type scan_number() // lgtm [cpp/use-of-goto] - { - // reset token_buffer to store the number's bytes - reset(); - - // the type of the parsed number; initially set to unsigned; will be - // changed if minus sign, decimal point or exponent is read - token_type number_type = token_type::value_unsigned; - - // state (init): we just found out we need to scan a number - switch (current) - { - case '-': - { - add(current); - goto scan_number_minus; - } - - case '0': - { - add(current); - goto scan_number_zero; - } - - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any1; - } - - // LCOV_EXCL_START - default: - { - // all other characters are rejected outside scan_number() - assert(false); - } - // LCOV_EXCL_STOP - } - -scan_number_minus: - // state: we just parsed a leading minus sign - number_type = token_type::value_integer; - switch (get()) - { - case '0': - { - add(current); - goto scan_number_zero; - } - - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any1; - } - - default: - { - error_message = "invalid number; expected digit after '-'"; - return token_type::parse_error; - } - } - -scan_number_zero: - // state: we just parse a zero (maybe with a leading minus sign) - switch (get()) - { - case '.': - { - add(decimal_point_char); - goto scan_number_decimal1; - } - - case 'e': - case 'E': - { - add(current); - goto scan_number_exponent; - } - - default: - goto scan_number_done; - } - -scan_number_any1: - // state: we just parsed a number 0-9 (maybe with a leading minus sign) - switch (get()) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any1; - } - - case '.': - { - add(decimal_point_char); - goto scan_number_decimal1; - } - - case 'e': - case 'E': - { - add(current); - goto scan_number_exponent; - } - - default: - goto scan_number_done; - } - -scan_number_decimal1: - // state: we just parsed a decimal point - number_type = token_type::value_float; - switch (get()) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_decimal2; - } - - default: - { - error_message = "invalid number; expected digit after '.'"; - return token_type::parse_error; - } - } - -scan_number_decimal2: - // we just parsed at least one number after a decimal point - switch (get()) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_decimal2; - } - - case 'e': - case 'E': - { - add(current); - goto scan_number_exponent; - } - - default: - goto scan_number_done; - } - -scan_number_exponent: - // we just parsed an exponent - number_type = token_type::value_float; - switch (get()) - { - case '+': - case '-': - { - add(current); - goto scan_number_sign; - } - - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any2; - } - - default: - { - error_message = - "invalid number; expected '+', '-', or digit after exponent"; - return token_type::parse_error; - } - } - -scan_number_sign: - // we just parsed an exponent sign - switch (get()) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any2; - } - - default: - { - error_message = "invalid number; expected digit after exponent sign"; - return token_type::parse_error; - } - } - -scan_number_any2: - // we just parsed a number after the exponent or exponent sign - switch (get()) - { - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - { - add(current); - goto scan_number_any2; - } - - default: - goto scan_number_done; - } - -scan_number_done: - // unget the character after the number (we only read it to know that - // we are done scanning a number) - unget(); - - char* endptr = nullptr; - errno = 0; - - // try to parse integers first and fall back to floats - if (number_type == token_type::value_unsigned) - { - const auto x = std::strtoull(token_buffer.data(), &endptr, 10); - - // we checked the number format before - assert(endptr == token_buffer.data() + token_buffer.size()); - - if (errno == 0) - { - value_unsigned = static_cast(x); - if (value_unsigned == x) - { - return token_type::value_unsigned; - } - } - } - else if (number_type == token_type::value_integer) - { - const auto x = std::strtoll(token_buffer.data(), &endptr, 10); - - // we checked the number format before - assert(endptr == token_buffer.data() + token_buffer.size()); - - if (errno == 0) - { - value_integer = static_cast(x); - if (value_integer == x) - { - return token_type::value_integer; - } - } - } - - // this code is reached if we parse a floating-point number or if an - // integer conversion above failed - strtof(value_float, token_buffer.data(), &endptr); - - // we checked the number format before - assert(endptr == token_buffer.data() + token_buffer.size()); - - return token_type::value_float; - } - - /*! - @param[in] literal_text the literal text to expect - @param[in] length the length of the passed literal text - @param[in] return_type the token type to return on success - */ - token_type scan_literal(const char* literal_text, const std::size_t length, - token_type return_type) - { - assert(current == literal_text[0]); - for (std::size_t i = 1; i < length; ++i) - { - if (JSON_UNLIKELY(get() != literal_text[i])) - { - error_message = "invalid literal"; - return token_type::parse_error; - } - } - return return_type; - } - - ///////////////////// - // input management - ///////////////////// - - /// reset token_buffer; current character is beginning of token - void reset() noexcept - { - token_buffer.clear(); - token_string.clear(); - token_string.push_back(std::char_traits::to_char_type(current)); - } - - /* - @brief get next character from the input - - This function provides the interface to the used input adapter. It does - not throw in case the input reached EOF, but returns a - `std::char_traits::eof()` in that case. Stores the scanned characters - for use in error messages. - - @return character read from the input - */ - std::char_traits::int_type get() - { - ++position.chars_read_total; - ++position.chars_read_current_line; - - if (next_unget) - { - // just reset the next_unget variable and work with current - next_unget = false; - } - else - { - current = ia->get_character(); - } - - if (JSON_LIKELY(current != std::char_traits::eof())) - { - token_string.push_back(std::char_traits::to_char_type(current)); - } - - if (current == '\n') - { - ++position.lines_read; - position.chars_read_current_line = 0; - } - - return current; - } - - /*! - @brief unget current character (read it again on next get) - - We implement unget by setting variable next_unget to true. The input is not - changed - we just simulate ungetting by modifying chars_read_total, - chars_read_current_line, and token_string. The next call to get() will - behave as if the unget character is read again. - */ - void unget() - { - next_unget = true; - - --position.chars_read_total; - - // in case we "unget" a newline, we have to also decrement the lines_read - if (position.chars_read_current_line == 0) - { - if (position.lines_read > 0) - { - --position.lines_read; - } - } - else - { - --position.chars_read_current_line; - } - - if (JSON_LIKELY(current != std::char_traits::eof())) - { - assert(not token_string.empty()); - token_string.pop_back(); - } - } - - /// add a character to token_buffer - void add(int c) - { - token_buffer.push_back(std::char_traits::to_char_type(c)); - } - - public: - ///////////////////// - // value getters - ///////////////////// - - /// return integer value - constexpr number_integer_t get_number_integer() const noexcept - { - return value_integer; - } - - /// return unsigned integer value - constexpr number_unsigned_t get_number_unsigned() const noexcept - { - return value_unsigned; - } - - /// return floating-point value - constexpr number_float_t get_number_float() const noexcept - { - return value_float; - } - - /// return current string value (implicitly resets the token; useful only once) - string_t& get_string() - { - return token_buffer; - } - - ///////////////////// - // diagnostics - ///////////////////// - - /// return position of last read token - constexpr position_t get_position() const noexcept - { - return position; - } - - /// return the last read token (for errors only). Will never contain EOF - /// (an arbitrary value that is not a valid char value, often -1), because - /// 255 may legitimately occur. May contain NUL, which should be escaped. - std::string get_token_string() const - { - // escape control characters - std::string result; - for (const auto c : token_string) - { - if ('\x00' <= c and c <= '\x1F') - { - // escape control characters - std::array cs{}; - (std::snprintf)(cs.data(), cs.size(), "", static_cast(c)); - result += cs.data(); - } - else - { - // add character as is - result.push_back(c); - } - } - - return result; - } - - /// return syntax error message - constexpr const char* get_error_message() const noexcept - { - return error_message; - } - - ///////////////////// - // actual scanner - ///////////////////// - - /*! - @brief skip the UTF-8 byte order mark - @return true iff there is no BOM or the correct BOM has been skipped - */ - bool skip_bom() - { - if (get() == 0xEF) - { - // check if we completely parse the BOM - return get() == 0xBB and get() == 0xBF; - } - - // the first character is not the beginning of the BOM; unget it to - // process is later - unget(); - return true; - } - - token_type scan() - { - // initially, skip the BOM - if (position.chars_read_total == 0 and not skip_bom()) - { - error_message = "invalid BOM; must be 0xEF 0xBB 0xBF if given"; - return token_type::parse_error; - } - - // read next character and ignore whitespace - do - { - get(); - } - while (current == ' ' or current == '\t' or current == '\n' or current == '\r'); - - switch (current) - { - // structural characters - case '[': - return token_type::begin_array; - case ']': - return token_type::end_array; - case '{': - return token_type::begin_object; - case '}': - return token_type::end_object; - case ':': - return token_type::name_separator; - case ',': - return token_type::value_separator; - - // literals - case 't': - return scan_literal("true", 4, token_type::literal_true); - case 'f': - return scan_literal("false", 5, token_type::literal_false); - case 'n': - return scan_literal("null", 4, token_type::literal_null); - - // string - case '\"': - return scan_string(); - - // number - case '-': - case '0': - case '1': - case '2': - case '3': - case '4': - case '5': - case '6': - case '7': - case '8': - case '9': - return scan_number(); - - // end of input (the null byte is needed when parsing from - // string literals) - case '\0': - case std::char_traits::eof(): - return token_type::end_of_input; - - // error - default: - error_message = "invalid literal"; - return token_type::parse_error; - } - } - - private: - /// input adapter - detail::input_adapter_t ia = nullptr; - - /// the current character - std::char_traits::int_type current = std::char_traits::eof(); - - /// whether the next get() call should just return current - bool next_unget = false; - - /// the start position of the current token - position_t position; - - /// raw input token string (for error messages) - std::vector token_string {}; - - /// buffer for variable-length tokens (numbers, strings) - string_t token_buffer {}; - - /// a description of occurred lexer errors - const char* error_message = ""; - - // number values - number_integer_t value_integer = 0; - number_unsigned_t value_unsigned = 0; - number_float_t value_float = 0; - - /// the decimal point - const char decimal_point_char = '.'; -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -#include // assert -#include // isfinite -#include // uint8_t -#include // function -#include // string -#include // move -#include // vector - -// #include - -// #include - -// #include - - -#include // size_t -#include // declval -#include // string - -// #include - -// #include - - -namespace nlohmann -{ -namespace detail -{ -template -using null_function_t = decltype(std::declval().null()); - -template -using boolean_function_t = - decltype(std::declval().boolean(std::declval())); - -template -using number_integer_function_t = - decltype(std::declval().number_integer(std::declval())); - -template -using number_unsigned_function_t = - decltype(std::declval().number_unsigned(std::declval())); - -template -using number_float_function_t = decltype(std::declval().number_float( - std::declval(), std::declval())); - -template -using string_function_t = - decltype(std::declval().string(std::declval())); - -template -using start_object_function_t = - decltype(std::declval().start_object(std::declval())); - -template -using key_function_t = - decltype(std::declval().key(std::declval())); - -template -using end_object_function_t = decltype(std::declval().end_object()); - -template -using start_array_function_t = - decltype(std::declval().start_array(std::declval())); - -template -using end_array_function_t = decltype(std::declval().end_array()); - -template -using parse_error_function_t = decltype(std::declval().parse_error( - std::declval(), std::declval(), - std::declval())); - -template -struct is_sax -{ - private: - static_assert(is_basic_json::value, - "BasicJsonType must be of type basic_json<...>"); - - using number_integer_t = typename BasicJsonType::number_integer_t; - using number_unsigned_t = typename BasicJsonType::number_unsigned_t; - using number_float_t = typename BasicJsonType::number_float_t; - using string_t = typename BasicJsonType::string_t; - using exception_t = typename BasicJsonType::exception; - - public: - static constexpr bool value = - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value && - is_detected_exact::value; -}; - -template -struct is_sax_static_asserts -{ - private: - static_assert(is_basic_json::value, - "BasicJsonType must be of type basic_json<...>"); - - using number_integer_t = typename BasicJsonType::number_integer_t; - using number_unsigned_t = typename BasicJsonType::number_unsigned_t; - using number_float_t = typename BasicJsonType::number_float_t; - using string_t = typename BasicJsonType::string_t; - using exception_t = typename BasicJsonType::exception; - - public: - static_assert(is_detected_exact::value, - "Missing/invalid function: bool null()"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool boolean(bool)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool boolean(bool)"); - static_assert( - is_detected_exact::value, - "Missing/invalid function: bool number_integer(number_integer_t)"); - static_assert( - is_detected_exact::value, - "Missing/invalid function: bool number_unsigned(number_unsigned_t)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool number_float(number_float_t, const string_t&)"); - static_assert( - is_detected_exact::value, - "Missing/invalid function: bool string(string_t&)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool start_object(std::size_t)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool key(string_t&)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool end_object()"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool start_array(std::size_t)"); - static_assert(is_detected_exact::value, - "Missing/invalid function: bool end_array()"); - static_assert( - is_detected_exact::value, - "Missing/invalid function: bool parse_error(std::size_t, const " - "std::string&, const exception&)"); -}; -} // namespace detail -} // namespace nlohmann - -// #include - // #include @@ -4442,6 +2837,13 @@ class json_sax_dom_parser : root(r), allow_exceptions(allow_exceptions_) {} + // make class move-only + json_sax_dom_parser(const json_sax_dom_parser&) = delete; + json_sax_dom_parser(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser& operator=(const json_sax_dom_parser&) = delete; + json_sax_dom_parser& operator=(json_sax_dom_parser&&) noexcept = default; + ~json_sax_dom_parser() = default; + bool null() { handle_value(nullptr); @@ -4590,7 +2992,7 @@ class json_sax_dom_parser /// the parsed JSON value BasicJsonType& root; /// stack to model hierarchy of values - std::vector ref_stack; + std::vector ref_stack {}; /// helper to hold the reference for the next object element BasicJsonType* object_element = nullptr; /// whether a syntax error occurred @@ -4618,6 +3020,13 @@ class json_sax_dom_callback_parser keep_stack.push_back(true); } + // make class move-only + json_sax_dom_callback_parser(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser(json_sax_dom_callback_parser&&) = default; + json_sax_dom_callback_parser& operator=(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser& operator=(json_sax_dom_callback_parser&&) = default; + ~json_sax_dom_callback_parser() = default; + bool null() { handle_value(nullptr); @@ -4895,11 +3304,11 @@ class json_sax_dom_callback_parser /// the parsed JSON value BasicJsonType& root; /// stack to model hierarchy of values - std::vector ref_stack; + std::vector ref_stack {}; /// stack to manage which values to keep - std::vector keep_stack; + std::vector keep_stack {}; /// stack to manage which object keys to keep - std::vector key_keep_stack; + std::vector key_keep_stack {}; /// helper to hold the reference for the next object element BasicJsonType* object_element = nullptr; /// whether a syntax error occurred @@ -4985,1540 +3394,154 @@ class json_sax_acceptor } // namespace nlohmann -// #include +// #include -// #include +// #include + + +#include // size_t +#include // declval +#include // string + +// #include + +// #include namespace nlohmann { namespace detail { -//////////// -// parser // -//////////// +template +using null_function_t = decltype(std::declval().null()); -/*! -@brief syntax analysis +template +using boolean_function_t = + decltype(std::declval().boolean(std::declval())); -This class implements a recursive decent parser. -*/ -template -class parser +template +using number_integer_function_t = + decltype(std::declval().number_integer(std::declval())); + +template +using number_unsigned_function_t = + decltype(std::declval().number_unsigned(std::declval())); + +template +using number_float_function_t = decltype(std::declval().number_float( + std::declval(), std::declval())); + +template +using string_function_t = + decltype(std::declval().string(std::declval())); + +template +using start_object_function_t = + decltype(std::declval().start_object(std::declval())); + +template +using key_function_t = + decltype(std::declval().key(std::declval())); + +template +using end_object_function_t = decltype(std::declval().end_object()); + +template +using start_array_function_t = + decltype(std::declval().start_array(std::declval())); + +template +using end_array_function_t = decltype(std::declval().end_array()); + +template +using parse_error_function_t = decltype(std::declval().parse_error( + std::declval(), std::declval(), + std::declval())); + +template +struct is_sax { + private: + static_assert(is_basic_json::value, + "BasicJsonType must be of type basic_json<...>"); + using number_integer_t = typename BasicJsonType::number_integer_t; using number_unsigned_t = typename BasicJsonType::number_unsigned_t; using number_float_t = typename BasicJsonType::number_float_t; using string_t = typename BasicJsonType::string_t; - using lexer_t = lexer; - using token_type = typename lexer_t::token_type; + using exception_t = typename BasicJsonType::exception; public: - enum class parse_event_t : uint8_t - { - /// the parser read `{` and started to process a JSON object - object_start, - /// the parser read `}` and finished processing a JSON object - object_end, - /// the parser read `[` and started to process a JSON array - array_start, - /// the parser read `]` and finished processing a JSON array - array_end, - /// the parser read a key of a value in an object - key, - /// the parser finished reading a JSON value - value - }; - - using parser_callback_t = - std::function; - - /// a parser reading from an input adapter - explicit parser(detail::input_adapter_t&& adapter, - const parser_callback_t cb = nullptr, - const bool allow_exceptions_ = true) - : callback(cb), m_lexer(std::move(adapter)), allow_exceptions(allow_exceptions_) - { - // read first token - get_token(); - } - - /*! - @brief public parser interface - - @param[in] strict whether to expect the last token to be EOF - @param[in,out] result parsed JSON value - - @throw parse_error.101 in case of an unexpected token - @throw parse_error.102 if to_unicode fails or surrogate error - @throw parse_error.103 if to_unicode fails - */ - void parse(const bool strict, BasicJsonType& result) - { - if (callback) - { - json_sax_dom_callback_parser sdp(result, callback, allow_exceptions); - sax_parse_internal(&sdp); - result.assert_invariant(); - - // in strict mode, input must be completely read - if (strict and (get_token() != token_type::end_of_input)) - { - sdp.parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_of_input, "value"))); - } - - // in case of an error, return discarded value - if (sdp.is_errored()) - { - result = value_t::discarded; - return; - } - - // set top-level value to null if it was discarded by the callback - // function - if (result.is_discarded()) - { - result = nullptr; - } - } - else - { - json_sax_dom_parser sdp(result, allow_exceptions); - sax_parse_internal(&sdp); - result.assert_invariant(); - - // in strict mode, input must be completely read - if (strict and (get_token() != token_type::end_of_input)) - { - sdp.parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_of_input, "value"))); - } - - // in case of an error, return discarded value - if (sdp.is_errored()) - { - result = value_t::discarded; - return; - } - } - } - - /*! - @brief public accept interface - - @param[in] strict whether to expect the last token to be EOF - @return whether the input is a proper JSON text - */ - bool accept(const bool strict = true) - { - json_sax_acceptor sax_acceptor; - return sax_parse(&sax_acceptor, strict); - } - - template - bool sax_parse(SAX* sax, const bool strict = true) - { - (void)detail::is_sax_static_asserts {}; - const bool result = sax_parse_internal(sax); - - // strict mode: next byte must be EOF - if (result and strict and (get_token() != token_type::end_of_input)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_of_input, "value"))); - } - - return result; - } + static constexpr bool value = + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value; +}; +template +struct is_sax_static_asserts +{ private: - template - bool sax_parse_internal(SAX* sax) - { - // stack to remember the hierarchy of structured values we are parsing - // true = array; false = object - std::vector states; - // value to avoid a goto (see comment where set to true) - bool skip_to_state_evaluation = false; + static_assert(is_basic_json::value, + "BasicJsonType must be of type basic_json<...>"); - while (true) - { - if (not skip_to_state_evaluation) - { - // invariant: get_token() was called before each iteration - switch (last_token) - { - case token_type::begin_object: - { - if (JSON_UNLIKELY(not sax->start_object(std::size_t(-1)))) - { - return false; - } + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using exception_t = typename BasicJsonType::exception; - // closing } -> we are done - if (get_token() == token_type::end_object) - { - if (JSON_UNLIKELY(not sax->end_object())) - { - return false; - } - break; - } - - // parse key - if (JSON_UNLIKELY(last_token != token_type::value_string)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::value_string, "object key"))); - } - if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) - { - return false; - } - - // parse separator (:) - if (JSON_UNLIKELY(get_token() != token_type::name_separator)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::name_separator, "object separator"))); - } - - // remember we are now inside an object - states.push_back(false); - - // parse values - get_token(); - continue; - } - - case token_type::begin_array: - { - if (JSON_UNLIKELY(not sax->start_array(std::size_t(-1)))) - { - return false; - } - - // closing ] -> we are done - if (get_token() == token_type::end_array) - { - if (JSON_UNLIKELY(not sax->end_array())) - { - return false; - } - break; - } - - // remember we are now inside an array - states.push_back(true); - - // parse values (no need to call get_token) - continue; - } - - case token_type::value_float: - { - const auto res = m_lexer.get_number_float(); - - if (JSON_UNLIKELY(not std::isfinite(res))) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); - } - else - { - if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) - { - return false; - } - break; - } - } - - case token_type::literal_false: - { - if (JSON_UNLIKELY(not sax->boolean(false))) - { - return false; - } - break; - } - - case token_type::literal_null: - { - if (JSON_UNLIKELY(not sax->null())) - { - return false; - } - break; - } - - case token_type::literal_true: - { - if (JSON_UNLIKELY(not sax->boolean(true))) - { - return false; - } - break; - } - - case token_type::value_integer: - { - if (JSON_UNLIKELY(not sax->number_integer(m_lexer.get_number_integer()))) - { - return false; - } - break; - } - - case token_type::value_string: - { - if (JSON_UNLIKELY(not sax->string(m_lexer.get_string()))) - { - return false; - } - break; - } - - case token_type::value_unsigned: - { - if (JSON_UNLIKELY(not sax->number_unsigned(m_lexer.get_number_unsigned()))) - { - return false; - } - break; - } - - case token_type::parse_error: - { - // using "uninitialized" to avoid "expected" message - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::uninitialized, "value"))); - } - - default: // the last token was unexpected - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::literal_or_value, "value"))); - } - } - } - else - { - skip_to_state_evaluation = false; - } - - // we reached this line after we successfully parsed a value - if (states.empty()) - { - // empty stack: we reached the end of the hierarchy: done - return true; - } - else - { - if (states.back()) // array - { - // comma -> next value - if (get_token() == token_type::value_separator) - { - // parse a new value - get_token(); - continue; - } - - // closing ] - if (JSON_LIKELY(last_token == token_type::end_array)) - { - if (JSON_UNLIKELY(not sax->end_array())) - { - return false; - } - - // We are done with this array. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_array, "array"))); - } - } - else // object - { - // comma -> next value - if (get_token() == token_type::value_separator) - { - // parse key - if (JSON_UNLIKELY(get_token() != token_type::value_string)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::value_string, "object key"))); - } - else - { - if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) - { - return false; - } - } - - // parse separator (:) - if (JSON_UNLIKELY(get_token() != token_type::name_separator)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::name_separator, "object separator"))); - } - - // parse values - get_token(); - continue; - } - - // closing } - if (JSON_LIKELY(last_token == token_type::end_object)) - { - if (JSON_UNLIKELY(not sax->end_object())) - { - return false; - } - - // We are done with this object. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_object, "object"))); - } - } - } - } - } - - /// get next token from lexer - token_type get_token() - { - return (last_token = m_lexer.scan()); - } - - std::string exception_message(const token_type expected, const std::string& context) - { - std::string error_msg = "syntax error "; - - if (not context.empty()) - { - error_msg += "while parsing " + context + " "; - } - - error_msg += "- "; - - if (last_token == token_type::parse_error) - { - error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + - m_lexer.get_token_string() + "'"; - } - else - { - error_msg += "unexpected " + std::string(lexer_t::token_type_name(last_token)); - } - - if (expected != token_type::uninitialized) - { - error_msg += "; expected " + std::string(lexer_t::token_type_name(expected)); - } - - return error_msg; - } - - private: - /// callback function - const parser_callback_t callback = nullptr; - /// the type of the last read token - token_type last_token = token_type::uninitialized; - /// the lexer - lexer_t m_lexer; - /// whether to throw exceptions in case of errors - const bool allow_exceptions = true; + public: + static_assert(is_detected_exact::value, + "Missing/invalid function: bool null()"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool boolean(bool)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool boolean(bool)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool number_integer(number_integer_t)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool number_unsigned(number_unsigned_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool number_float(number_float_t, const string_t&)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool string(string_t&)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool start_object(std::size_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool key(string_t&)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool end_object()"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool start_array(std::size_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool end_array()"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool parse_error(std::size_t, const " + "std::string&, const exception&)"); }; } // namespace detail } // namespace nlohmann -// #include - - -#include // ptrdiff_t -#include // numeric_limits - -namespace nlohmann -{ -namespace detail -{ -/* -@brief an iterator for primitive JSON types - -This class models an iterator for primitive JSON types (boolean, number, -string). It's only purpose is to allow the iterator/const_iterator classes -to "iterate" over primitive values. Internally, the iterator is modeled by -a `difference_type` variable. Value begin_value (`0`) models the begin, -end_value (`1`) models past the end. -*/ -class primitive_iterator_t -{ - private: - using difference_type = std::ptrdiff_t; - static constexpr difference_type begin_value = 0; - static constexpr difference_type end_value = begin_value + 1; - - /// iterator as signed integer type - difference_type m_it = (std::numeric_limits::min)(); - - public: - constexpr difference_type get_value() const noexcept - { - return m_it; - } - - /// set iterator to a defined beginning - void set_begin() noexcept - { - m_it = begin_value; - } - - /// set iterator to a defined past the end - void set_end() noexcept - { - m_it = end_value; - } - - /// return whether the iterator can be dereferenced - constexpr bool is_begin() const noexcept - { - return m_it == begin_value; - } - - /// return whether the iterator is at end - constexpr bool is_end() const noexcept - { - return m_it == end_value; - } - - friend constexpr bool operator==(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it == rhs.m_it; - } - - friend constexpr bool operator<(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it < rhs.m_it; - } - - primitive_iterator_t operator+(difference_type n) noexcept - { - auto result = *this; - result += n; - return result; - } - - friend constexpr difference_type operator-(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept - { - return lhs.m_it - rhs.m_it; - } - - primitive_iterator_t& operator++() noexcept - { - ++m_it; - return *this; - } - - primitive_iterator_t const operator++(int) noexcept - { - auto result = *this; - ++m_it; - return result; - } - - primitive_iterator_t& operator--() noexcept - { - --m_it; - return *this; - } - - primitive_iterator_t const operator--(int) noexcept - { - auto result = *this; - --m_it; - return result; - } - - primitive_iterator_t& operator+=(difference_type n) noexcept - { - m_it += n; - return *this; - } - - primitive_iterator_t& operator-=(difference_type n) noexcept - { - m_it -= n; - return *this; - } -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -// #include - - -namespace nlohmann -{ -namespace detail -{ -/*! -@brief an iterator value - -@note This structure could easily be a union, but MSVC currently does not allow -unions members with complex constructors, see https://github.com/nlohmann/json/pull/105. -*/ -template struct internal_iterator -{ - /// iterator for JSON objects - typename BasicJsonType::object_t::iterator object_iterator {}; - /// iterator for JSON arrays - typename BasicJsonType::array_t::iterator array_iterator {}; - /// generic iterator for all other types - primitive_iterator_t primitive_iterator {}; -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -#include // not -#include // iterator, random_access_iterator_tag, bidirectional_iterator_tag, advance, next -#include // conditional, is_const, remove_const - -// #include - -// #include - -// #include - -// #include - -// #include - -// #include - -// #include - - -namespace nlohmann -{ -namespace detail -{ -// forward declare, to be able to friend it later on -template class iteration_proxy; -template class iteration_proxy_value; - -/*! -@brief a template for a bidirectional iterator for the @ref basic_json class -This class implements a both iterators (iterator and const_iterator) for the -@ref basic_json class. -@note An iterator is called *initialized* when a pointer to a JSON value has - been set (e.g., by a constructor or a copy assignment). If the iterator is - default-constructed, it is *uninitialized* and most methods are undefined. - **The library uses assertions to detect calls on uninitialized iterators.** -@requirement The class satisfies the following concept requirements: -- -[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): - The iterator that can be moved can be moved in both directions (i.e. - incremented and decremented). -@since version 1.0.0, simplified in version 2.0.9, change to bidirectional - iterators in version 3.0.0 (see https://github.com/nlohmann/json/issues/593) -*/ -template -class iter_impl -{ - /// allow basic_json to access private members - friend iter_impl::value, typename std::remove_const::type, const BasicJsonType>::type>; - friend BasicJsonType; - friend iteration_proxy; - friend iteration_proxy_value; - - using object_t = typename BasicJsonType::object_t; - using array_t = typename BasicJsonType::array_t; - // make sure BasicJsonType is basic_json or const basic_json - static_assert(is_basic_json::type>::value, - "iter_impl only accepts (const) basic_json"); - - public: - - /// The std::iterator class template (used as a base class to provide typedefs) is deprecated in C++17. - /// The C++ Standard has never required user-defined iterators to derive from std::iterator. - /// A user-defined iterator should provide publicly accessible typedefs named - /// iterator_category, value_type, difference_type, pointer, and reference. - /// Note that value_type is required to be non-const, even for constant iterators. - using iterator_category = std::bidirectional_iterator_tag; - - /// the type of the values when the iterator is dereferenced - using value_type = typename BasicJsonType::value_type; - /// a type to represent differences between iterators - using difference_type = typename BasicJsonType::difference_type; - /// defines a pointer to the type iterated over (value_type) - using pointer = typename std::conditional::value, - typename BasicJsonType::const_pointer, - typename BasicJsonType::pointer>::type; - /// defines a reference to the type iterated over (value_type) - using reference = - typename std::conditional::value, - typename BasicJsonType::const_reference, - typename BasicJsonType::reference>::type; - - /// default constructor - iter_impl() = default; - - /*! - @brief constructor for a given JSON instance - @param[in] object pointer to a JSON object for this iterator - @pre object != nullptr - @post The iterator is initialized; i.e. `m_object != nullptr`. - */ - explicit iter_impl(pointer object) noexcept : m_object(object) - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - m_it.object_iterator = typename object_t::iterator(); - break; - } - - case value_t::array: - { - m_it.array_iterator = typename array_t::iterator(); - break; - } - - default: - { - m_it.primitive_iterator = primitive_iterator_t(); - break; - } - } - } - - /*! - @note The conventional copy constructor and copy assignment are implicitly - defined. Combined with the following converting constructor and - assignment, they support: (1) copy from iterator to iterator, (2) - copy from const iterator to const iterator, and (3) conversion from - iterator to const iterator. However conversion from const iterator - to iterator is not defined. - */ - - /*! - @brief converting constructor - @param[in] other non-const iterator to copy from - @note It is not checked whether @a other is initialized. - */ - iter_impl(const iter_impl::type>& other) noexcept - : m_object(other.m_object), m_it(other.m_it) {} - - /*! - @brief converting assignment - @param[in,out] other non-const iterator to copy from - @return const/non-const iterator - @note It is not checked whether @a other is initialized. - */ - iter_impl& operator=(const iter_impl::type>& other) noexcept - { - m_object = other.m_object; - m_it = other.m_it; - return *this; - } - - private: - /*! - @brief set the iterator to the first value - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - void set_begin() noexcept - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - m_it.object_iterator = m_object->m_value.object->begin(); - break; - } - - case value_t::array: - { - m_it.array_iterator = m_object->m_value.array->begin(); - break; - } - - case value_t::null: - { - // set to end so begin()==end() is true: null is empty - m_it.primitive_iterator.set_end(); - break; - } - - default: - { - m_it.primitive_iterator.set_begin(); - break; - } - } - } - - /*! - @brief set the iterator past the last value - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - void set_end() noexcept - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - m_it.object_iterator = m_object->m_value.object->end(); - break; - } - - case value_t::array: - { - m_it.array_iterator = m_object->m_value.array->end(); - break; - } - - default: - { - m_it.primitive_iterator.set_end(); - break; - } - } - } - - public: - /*! - @brief return a reference to the value pointed to by the iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference operator*() const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - assert(m_it.object_iterator != m_object->m_value.object->end()); - return m_it.object_iterator->second; - } - - case value_t::array: - { - assert(m_it.array_iterator != m_object->m_value.array->end()); - return *m_it.array_iterator; - } - - case value_t::null: - JSON_THROW(invalid_iterator::create(214, "cannot get value")); - - default: - { - if (JSON_LIKELY(m_it.primitive_iterator.is_begin())) - { - return *m_object; - } - - JSON_THROW(invalid_iterator::create(214, "cannot get value")); - } - } - } - - /*! - @brief dereference the iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - pointer operator->() const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - assert(m_it.object_iterator != m_object->m_value.object->end()); - return &(m_it.object_iterator->second); - } - - case value_t::array: - { - assert(m_it.array_iterator != m_object->m_value.array->end()); - return &*m_it.array_iterator; - } - - default: - { - if (JSON_LIKELY(m_it.primitive_iterator.is_begin())) - { - return m_object; - } - - JSON_THROW(invalid_iterator::create(214, "cannot get value")); - } - } - } - - /*! - @brief post-increment (it++) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl const operator++(int) - { - auto result = *this; - ++(*this); - return result; - } - - /*! - @brief pre-increment (++it) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator++() - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - std::advance(m_it.object_iterator, 1); - break; - } - - case value_t::array: - { - std::advance(m_it.array_iterator, 1); - break; - } - - default: - { - ++m_it.primitive_iterator; - break; - } - } - - return *this; - } - - /*! - @brief post-decrement (it--) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl const operator--(int) - { - auto result = *this; - --(*this); - return result; - } - - /*! - @brief pre-decrement (--it) - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator--() - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - { - std::advance(m_it.object_iterator, -1); - break; - } - - case value_t::array: - { - std::advance(m_it.array_iterator, -1); - break; - } - - default: - { - --m_it.primitive_iterator; - break; - } - } - - return *this; - } - - /*! - @brief comparison: equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator==(const iter_impl& other) const - { - // if objects are not the same, the comparison is undefined - if (JSON_UNLIKELY(m_object != other.m_object)) - { - JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); - } - - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - return (m_it.object_iterator == other.m_it.object_iterator); - - case value_t::array: - return (m_it.array_iterator == other.m_it.array_iterator); - - default: - return (m_it.primitive_iterator == other.m_it.primitive_iterator); - } - } - - /*! - @brief comparison: not equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator!=(const iter_impl& other) const - { - return not operator==(other); - } - - /*! - @brief comparison: smaller - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator<(const iter_impl& other) const - { - // if objects are not the same, the comparison is undefined - if (JSON_UNLIKELY(m_object != other.m_object)) - { - JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); - } - - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - JSON_THROW(invalid_iterator::create(213, "cannot compare order of object iterators")); - - case value_t::array: - return (m_it.array_iterator < other.m_it.array_iterator); - - default: - return (m_it.primitive_iterator < other.m_it.primitive_iterator); - } - } - - /*! - @brief comparison: less than or equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator<=(const iter_impl& other) const - { - return not other.operator < (*this); - } - - /*! - @brief comparison: greater than - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator>(const iter_impl& other) const - { - return not operator<=(other); - } - - /*! - @brief comparison: greater than or equal - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - bool operator>=(const iter_impl& other) const - { - return not operator<(other); - } - - /*! - @brief add to iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator+=(difference_type i) - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); - - case value_t::array: - { - std::advance(m_it.array_iterator, i); - break; - } - - default: - { - m_it.primitive_iterator += i; - break; - } - } - - return *this; - } - - /*! - @brief subtract from iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl& operator-=(difference_type i) - { - return operator+=(-i); - } - - /*! - @brief add to iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator+(difference_type i) const - { - auto result = *this; - result += i; - return result; - } - - /*! - @brief addition of distance and iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - friend iter_impl operator+(difference_type i, const iter_impl& it) - { - auto result = it; - result += i; - return result; - } - - /*! - @brief subtract from iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - iter_impl operator-(difference_type i) const - { - auto result = *this; - result -= i; - return result; - } - - /*! - @brief return difference - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - difference_type operator-(const iter_impl& other) const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); - - case value_t::array: - return m_it.array_iterator - other.m_it.array_iterator; - - default: - return m_it.primitive_iterator - other.m_it.primitive_iterator; - } - } - - /*! - @brief access to successor - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference operator[](difference_type n) const - { - assert(m_object != nullptr); - - switch (m_object->m_type) - { - case value_t::object: - JSON_THROW(invalid_iterator::create(208, "cannot use operator[] for object iterators")); - - case value_t::array: - return *std::next(m_it.array_iterator, n); - - case value_t::null: - JSON_THROW(invalid_iterator::create(214, "cannot get value")); - - default: - { - if (JSON_LIKELY(m_it.primitive_iterator.get_value() == -n)) - { - return *m_object; - } - - JSON_THROW(invalid_iterator::create(214, "cannot get value")); - } - } - } - - /*! - @brief return the key of an object iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - const typename object_t::key_type& key() const - { - assert(m_object != nullptr); - - if (JSON_LIKELY(m_object->is_object())) - { - return m_it.object_iterator->first; - } - - JSON_THROW(invalid_iterator::create(207, "cannot use key() for non-object iterators")); - } - - /*! - @brief return the value of an iterator - @pre The iterator is initialized; i.e. `m_object != nullptr`. - */ - reference value() const - { - return operator*(); - } - - private: - /// associated JSON instance - pointer m_object = nullptr; - /// the actual iterator of the associated instance - internal_iterator::type> m_it; -}; -} // namespace detail -} // namespace nlohmann -// #include - -// #include - - -#include // ptrdiff_t -#include // reverse_iterator -#include // declval - -namespace nlohmann -{ -namespace detail -{ -////////////////////// -// reverse_iterator // -////////////////////// - -/*! -@brief a template for a reverse iterator class - -@tparam Base the base iterator type to reverse. Valid types are @ref -iterator (to create @ref reverse_iterator) and @ref const_iterator (to -create @ref const_reverse_iterator). - -@requirement The class satisfies the following concept requirements: -- -[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): - The iterator that can be moved can be moved in both directions (i.e. - incremented and decremented). -- [OutputIterator](https://en.cppreference.com/w/cpp/named_req/OutputIterator): - It is possible to write to the pointed-to element (only if @a Base is - @ref iterator). - -@since version 1.0.0 -*/ -template -class json_reverse_iterator : public std::reverse_iterator -{ - public: - using difference_type = std::ptrdiff_t; - /// shortcut to the reverse iterator adapter - using base_iterator = std::reverse_iterator; - /// the reference type for the pointed-to element - using reference = typename Base::reference; - - /// create reverse iterator from iterator - explicit json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept - : base_iterator(it) {} - - /// create reverse iterator from base class - explicit json_reverse_iterator(const base_iterator& it) noexcept : base_iterator(it) {} - - /// post-increment (it++) - json_reverse_iterator const operator++(int) - { - return static_cast(base_iterator::operator++(1)); - } - - /// pre-increment (++it) - json_reverse_iterator& operator++() - { - return static_cast(base_iterator::operator++()); - } - - /// post-decrement (it--) - json_reverse_iterator const operator--(int) - { - return static_cast(base_iterator::operator--(1)); - } - - /// pre-decrement (--it) - json_reverse_iterator& operator--() - { - return static_cast(base_iterator::operator--()); - } - - /// add to iterator - json_reverse_iterator& operator+=(difference_type i) - { - return static_cast(base_iterator::operator+=(i)); - } - - /// add to iterator - json_reverse_iterator operator+(difference_type i) const - { - return static_cast(base_iterator::operator+(i)); - } - - /// subtract from iterator - json_reverse_iterator operator-(difference_type i) const - { - return static_cast(base_iterator::operator-(i)); - } - - /// return difference - difference_type operator-(const json_reverse_iterator& other) const - { - return base_iterator(*this) - base_iterator(other); - } - - /// access to successor - reference operator[](difference_type n) const - { - return *(this->operator+(n)); - } - - /// return the key of an object iterator - auto key() const -> decltype(std::declval().key()) - { - auto it = --this->base(); - return it.key(); - } - - /// return the value of an iterator - reference value() const - { - auto it = --this->base(); - return it.operator * (); - } -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -#include // copy -#include // size_t -#include // streamsize -#include // back_inserter -#include // shared_ptr, make_shared -#include // basic_ostream -#include // basic_string -#include // vector - -namespace nlohmann -{ -namespace detail -{ -/// abstract output adapter interface -template struct output_adapter_protocol -{ - virtual void write_character(CharType c) = 0; - virtual void write_characters(const CharType* s, std::size_t length) = 0; - virtual ~output_adapter_protocol() = default; -}; - -/// a type to simplify interfaces -template -using output_adapter_t = std::shared_ptr>; - -/// output adapter for byte vectors -template -class output_vector_adapter : public output_adapter_protocol -{ - public: - explicit output_vector_adapter(std::vector& vec) noexcept - : v(vec) - {} - - void write_character(CharType c) override - { - v.push_back(c); - } - - void write_characters(const CharType* s, std::size_t length) override - { - std::copy(s, s + length, std::back_inserter(v)); - } - - private: - std::vector& v; -}; - -/// output adapter for output streams -template -class output_stream_adapter : public output_adapter_protocol -{ - public: - explicit output_stream_adapter(std::basic_ostream& s) noexcept - : stream(s) - {} - - void write_character(CharType c) override - { - stream.put(c); - } - - void write_characters(const CharType* s, std::size_t length) override - { - stream.write(s, static_cast(length)); - } - - private: - std::basic_ostream& stream; -}; - -/// output adapter for basic_string -template> -class output_string_adapter : public output_adapter_protocol -{ - public: - explicit output_string_adapter(StringType& s) noexcept - : str(s) - {} - - void write_character(CharType c) override - { - str.push_back(c); - } - - void write_characters(const CharType* s, std::size_t length) override - { - str.append(s, length); - } - - private: - StringType& str; -}; - -template> -class output_adapter -{ - public: - output_adapter(std::vector& vec) - : oa(std::make_shared>(vec)) {} - - output_adapter(std::basic_ostream& s) - : oa(std::make_shared>(s)) {} - - output_adapter(StringType& s) - : oa(std::make_shared>(s)) {} - - operator output_adapter_t() - { - return oa; - } - - private: - output_adapter_t oa = nullptr; -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -#include // generate_n -#include // array -#include // assert -#include // ldexp -#include // size_t -#include // uint8_t, uint16_t, uint32_t, uint64_t -#include // snprintf -#include // memcpy -#include // back_inserter -#include // numeric_limits -#include // char_traits, string -#include // make_pair, move - -// #include - -// #include - -// #include - -// #include - -// #include - // #include @@ -6554,6 +3577,13 @@ class binary_reader assert(ia); } + // make class move-only + binary_reader(const binary_reader&) = delete; + binary_reader(binary_reader&&) noexcept = default; + binary_reader& operator=(const binary_reader&) = delete; + binary_reader& operator=(binary_reader&&) noexcept = default; + ~binary_reader() = default; + /*! @param[in] format the binary format to parse @param[in] sax_ a SAX event processor @@ -8487,6 +5517,3786 @@ class binary_reader } // namespace detail } // namespace nlohmann +// #include + +// #include + + +#include // array +#include // localeconv +#include // size_t +#include // snprintf +#include // strtof, strtod, strtold, strtoll, strtoull +#include // initializer_list +#include // char_traits, string +#include // move +#include // vector + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/////////// +// lexer // +/////////// + +/*! +@brief lexical analysis + +This class organizes the lexical analysis during JSON deserialization. +*/ +template +class lexer +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + + public: + /// token types for the parser + enum class token_type + { + uninitialized, ///< indicating the scanner is uninitialized + literal_true, ///< the `true` literal + literal_false, ///< the `false` literal + literal_null, ///< the `null` literal + value_string, ///< a string -- use get_string() for actual value + value_unsigned, ///< an unsigned integer -- use get_number_unsigned() for actual value + value_integer, ///< a signed integer -- use get_number_integer() for actual value + value_float, ///< an floating point number -- use get_number_float() for actual value + begin_array, ///< the character for array begin `[` + begin_object, ///< the character for object begin `{` + end_array, ///< the character for array end `]` + end_object, ///< the character for object end `}` + name_separator, ///< the name separator `:` + value_separator, ///< the value separator `,` + parse_error, ///< indicating a parse error + end_of_input, ///< indicating the end of the input buffer + literal_or_value ///< a literal or the begin of a value (only for diagnostics) + }; + + /// return name of values of type token_type (only used for errors) + static const char* token_type_name(const token_type t) noexcept + { + switch (t) + { + case token_type::uninitialized: + return ""; + case token_type::literal_true: + return "true literal"; + case token_type::literal_false: + return "false literal"; + case token_type::literal_null: + return "null literal"; + case token_type::value_string: + return "string literal"; + case lexer::token_type::value_unsigned: + case lexer::token_type::value_integer: + case lexer::token_type::value_float: + return "number literal"; + case token_type::begin_array: + return "'['"; + case token_type::begin_object: + return "'{'"; + case token_type::end_array: + return "']'"; + case token_type::end_object: + return "'}'"; + case token_type::name_separator: + return "':'"; + case token_type::value_separator: + return "','"; + case token_type::parse_error: + return ""; + case token_type::end_of_input: + return "end of input"; + case token_type::literal_or_value: + return "'[', '{', or a literal"; + // LCOV_EXCL_START + default: // catch non-enum values + return "unknown token"; + // LCOV_EXCL_STOP + } + } + + explicit lexer(detail::input_adapter_t&& adapter) + : ia(std::move(adapter)), decimal_point_char(get_decimal_point()) {} + + // delete because of pointer members + lexer(const lexer&) = delete; + lexer(lexer&&) = delete; + lexer& operator=(lexer&) = delete; + lexer& operator=(lexer&&) = delete; + ~lexer() = default; + + private: + ///////////////////// + // locales + ///////////////////// + + /// return the locale-dependent decimal point + static char get_decimal_point() noexcept + { + const auto loc = localeconv(); + assert(loc != nullptr); + return (loc->decimal_point == nullptr) ? '.' : *(loc->decimal_point); + } + + ///////////////////// + // scan functions + ///////////////////// + + /*! + @brief get codepoint from 4 hex characters following `\u` + + For input "\u c1 c2 c3 c4" the codepoint is: + (c1 * 0x1000) + (c2 * 0x0100) + (c3 * 0x0010) + c4 + = (c1 << 12) + (c2 << 8) + (c3 << 4) + (c4 << 0) + + Furthermore, the possible characters '0'..'9', 'A'..'F', and 'a'..'f' + must be converted to the integers 0x0..0x9, 0xA..0xF, 0xA..0xF, resp. The + conversion is done by subtracting the offset (0x30, 0x37, and 0x57) + between the ASCII value of the character and the desired integer value. + + @return codepoint (0x0000..0xFFFF) or -1 in case of an error (e.g. EOF or + non-hex character) + */ + int get_codepoint() + { + // this function only makes sense after reading `\u` + assert(current == 'u'); + int codepoint = 0; + + const auto factors = { 12u, 8u, 4u, 0u }; + for (const auto factor : factors) + { + get(); + + if (current >= '0' and current <= '9') + { + codepoint += static_cast((static_cast(current) - 0x30u) << factor); + } + else if (current >= 'A' and current <= 'F') + { + codepoint += static_cast((static_cast(current) - 0x37u) << factor); + } + else if (current >= 'a' and current <= 'f') + { + codepoint += static_cast((static_cast(current) - 0x57u) << factor); + } + else + { + return -1; + } + } + + assert(0x0000 <= codepoint and codepoint <= 0xFFFF); + return codepoint; + } + + /*! + @brief check if the next byte(s) are inside a given range + + Adds the current byte and, for each passed range, reads a new byte and + checks if it is inside the range. If a violation was detected, set up an + error message and return false. Otherwise, return true. + + @param[in] ranges list of integers; interpreted as list of pairs of + inclusive lower and upper bound, respectively + + @pre The passed list @a ranges must have 2, 4, or 6 elements; that is, + 1, 2, or 3 pairs. This precondition is enforced by an assertion. + + @return true if and only if no range violation was detected + */ + bool next_byte_in_range(std::initializer_list ranges) + { + assert(ranges.size() == 2 or ranges.size() == 4 or ranges.size() == 6); + add(current); + + for (auto range = ranges.begin(); range != ranges.end(); ++range) + { + get(); + if (JSON_LIKELY(*range <= current and current <= *(++range))) + { + add(current); + } + else + { + error_message = "invalid string: ill-formed UTF-8 byte"; + return false; + } + } + + return true; + } + + /*! + @brief scan a string literal + + This function scans a string according to Sect. 7 of RFC 7159. While + scanning, bytes are escaped and copied into buffer token_buffer. Then the + function returns successfully, token_buffer is *not* null-terminated (as it + may contain \0 bytes), and token_buffer.size() is the number of bytes in the + string. + + @return token_type::value_string if string could be successfully scanned, + token_type::parse_error otherwise + + @note In case of errors, variable error_message contains a textual + description. + */ + token_type scan_string() + { + // reset token_buffer (ignore opening quote) + reset(); + + // we entered the function by reading an open quote + assert(current == '\"'); + + while (true) + { + // get next character + switch (get()) + { + // end of file while parsing string + case std::char_traits::eof(): + { + error_message = "invalid string: missing closing quote"; + return token_type::parse_error; + } + + // closing quote + case '\"': + { + return token_type::value_string; + } + + // escapes + case '\\': + { + switch (get()) + { + // quotation mark + case '\"': + add('\"'); + break; + // reverse solidus + case '\\': + add('\\'); + break; + // solidus + case '/': + add('/'); + break; + // backspace + case 'b': + add('\b'); + break; + // form feed + case 'f': + add('\f'); + break; + // line feed + case 'n': + add('\n'); + break; + // carriage return + case 'r': + add('\r'); + break; + // tab + case 't': + add('\t'); + break; + + // unicode escapes + case 'u': + { + const int codepoint1 = get_codepoint(); + int codepoint = codepoint1; // start with codepoint1 + + if (JSON_UNLIKELY(codepoint1 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if code point is a high surrogate + if (0xD800 <= codepoint1 and codepoint1 <= 0xDBFF) + { + // expect next \uxxxx entry + if (JSON_LIKELY(get() == '\\' and get() == 'u')) + { + const int codepoint2 = get_codepoint(); + + if (JSON_UNLIKELY(codepoint2 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if codepoint2 is a low surrogate + if (JSON_LIKELY(0xDC00 <= codepoint2 and codepoint2 <= 0xDFFF)) + { + // overwrite codepoint + codepoint = static_cast( + // high surrogate occupies the most significant 22 bits + (static_cast(codepoint1) << 10u) + // low surrogate occupies the least significant 15 bits + + static_cast(codepoint2) + // there is still the 0xD800, 0xDC00 and 0x10000 noise + // in the result so we have to subtract with: + // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 + - 0x35FDC00u); + } + else + { + error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; + return token_type::parse_error; + } + } + else + { + error_message = "invalid string: surrogate U+DC00..U+DFFF must be followed by U+DC00..U+DFFF"; + return token_type::parse_error; + } + } + else + { + if (JSON_UNLIKELY(0xDC00 <= codepoint1 and codepoint1 <= 0xDFFF)) + { + error_message = "invalid string: surrogate U+DC00..U+DFFF must follow U+D800..U+DBFF"; + return token_type::parse_error; + } + } + + // result of the above calculation yields a proper codepoint + assert(0x00 <= codepoint and codepoint <= 0x10FFFF); + + // translate codepoint into bytes + if (codepoint < 0x80) + { + // 1-byte characters: 0xxxxxxx (ASCII) + add(codepoint); + } + else if (codepoint <= 0x7FF) + { + // 2-byte characters: 110xxxxx 10xxxxxx + add(static_cast(0xC0u | (static_cast(codepoint) >> 6u))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + else if (codepoint <= 0xFFFF) + { + // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx + add(static_cast(0xE0u | (static_cast(codepoint) >> 12u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + else + { + // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + add(static_cast(0xF0u | (static_cast(codepoint) >> 18u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + + break; + } + + // other characters after escape + default: + error_message = "invalid string: forbidden character after backslash"; + return token_type::parse_error; + } + + break; + } + + // invalid control characters + case 0x00: + { + error_message = "invalid string: control character U+0000 (NUL) must be escaped to \\u0000"; + return token_type::parse_error; + } + + case 0x01: + { + error_message = "invalid string: control character U+0001 (SOH) must be escaped to \\u0001"; + return token_type::parse_error; + } + + case 0x02: + { + error_message = "invalid string: control character U+0002 (STX) must be escaped to \\u0002"; + return token_type::parse_error; + } + + case 0x03: + { + error_message = "invalid string: control character U+0003 (ETX) must be escaped to \\u0003"; + return token_type::parse_error; + } + + case 0x04: + { + error_message = "invalid string: control character U+0004 (EOT) must be escaped to \\u0004"; + return token_type::parse_error; + } + + case 0x05: + { + error_message = "invalid string: control character U+0005 (ENQ) must be escaped to \\u0005"; + return token_type::parse_error; + } + + case 0x06: + { + error_message = "invalid string: control character U+0006 (ACK) must be escaped to \\u0006"; + return token_type::parse_error; + } + + case 0x07: + { + error_message = "invalid string: control character U+0007 (BEL) must be escaped to \\u0007"; + return token_type::parse_error; + } + + case 0x08: + { + error_message = "invalid string: control character U+0008 (BS) must be escaped to \\u0008 or \\b"; + return token_type::parse_error; + } + + case 0x09: + { + error_message = "invalid string: control character U+0009 (HT) must be escaped to \\u0009 or \\t"; + return token_type::parse_error; + } + + case 0x0A: + { + error_message = "invalid string: control character U+000A (LF) must be escaped to \\u000A or \\n"; + return token_type::parse_error; + } + + case 0x0B: + { + error_message = "invalid string: control character U+000B (VT) must be escaped to \\u000B"; + return token_type::parse_error; + } + + case 0x0C: + { + error_message = "invalid string: control character U+000C (FF) must be escaped to \\u000C or \\f"; + return token_type::parse_error; + } + + case 0x0D: + { + error_message = "invalid string: control character U+000D (CR) must be escaped to \\u000D or \\r"; + return token_type::parse_error; + } + + case 0x0E: + { + error_message = "invalid string: control character U+000E (SO) must be escaped to \\u000E"; + return token_type::parse_error; + } + + case 0x0F: + { + error_message = "invalid string: control character U+000F (SI) must be escaped to \\u000F"; + return token_type::parse_error; + } + + case 0x10: + { + error_message = "invalid string: control character U+0010 (DLE) must be escaped to \\u0010"; + return token_type::parse_error; + } + + case 0x11: + { + error_message = "invalid string: control character U+0011 (DC1) must be escaped to \\u0011"; + return token_type::parse_error; + } + + case 0x12: + { + error_message = "invalid string: control character U+0012 (DC2) must be escaped to \\u0012"; + return token_type::parse_error; + } + + case 0x13: + { + error_message = "invalid string: control character U+0013 (DC3) must be escaped to \\u0013"; + return token_type::parse_error; + } + + case 0x14: + { + error_message = "invalid string: control character U+0014 (DC4) must be escaped to \\u0014"; + return token_type::parse_error; + } + + case 0x15: + { + error_message = "invalid string: control character U+0015 (NAK) must be escaped to \\u0015"; + return token_type::parse_error; + } + + case 0x16: + { + error_message = "invalid string: control character U+0016 (SYN) must be escaped to \\u0016"; + return token_type::parse_error; + } + + case 0x17: + { + error_message = "invalid string: control character U+0017 (ETB) must be escaped to \\u0017"; + return token_type::parse_error; + } + + case 0x18: + { + error_message = "invalid string: control character U+0018 (CAN) must be escaped to \\u0018"; + return token_type::parse_error; + } + + case 0x19: + { + error_message = "invalid string: control character U+0019 (EM) must be escaped to \\u0019"; + return token_type::parse_error; + } + + case 0x1A: + { + error_message = "invalid string: control character U+001A (SUB) must be escaped to \\u001A"; + return token_type::parse_error; + } + + case 0x1B: + { + error_message = "invalid string: control character U+001B (ESC) must be escaped to \\u001B"; + return token_type::parse_error; + } + + case 0x1C: + { + error_message = "invalid string: control character U+001C (FS) must be escaped to \\u001C"; + return token_type::parse_error; + } + + case 0x1D: + { + error_message = "invalid string: control character U+001D (GS) must be escaped to \\u001D"; + return token_type::parse_error; + } + + case 0x1E: + { + error_message = "invalid string: control character U+001E (RS) must be escaped to \\u001E"; + return token_type::parse_error; + } + + case 0x1F: + { + error_message = "invalid string: control character U+001F (US) must be escaped to \\u001F"; + return token_type::parse_error; + } + + // U+0020..U+007F (except U+0022 (quote) and U+005C (backspace)) + case 0x20: + case 0x21: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2A: + case 0x2B: + case 0x2C: + case 0x2D: + case 0x2E: + case 0x2F: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3A: + case 0x3B: + case 0x3C: + case 0x3D: + case 0x3E: + case 0x3F: + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4A: + case 0x4B: + case 0x4C: + case 0x4D: + case 0x4E: + case 0x4F: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: + case 0x59: + case 0x5A: + case 0x5B: + case 0x5D: + case 0x5E: + case 0x5F: + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6A: + case 0x6B: + case 0x6C: + case 0x6D: + case 0x6E: + case 0x6F: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: + case 0x79: + case 0x7A: + case 0x7B: + case 0x7C: + case 0x7D: + case 0x7E: + case 0x7F: + { + add(current); + break; + } + + // U+0080..U+07FF: bytes C2..DF 80..BF + case 0xC2: + case 0xC3: + case 0xC4: + case 0xC5: + case 0xC6: + case 0xC7: + case 0xC8: + case 0xC9: + case 0xCA: + case 0xCB: + case 0xCC: + case 0xCD: + case 0xCE: + case 0xCF: + case 0xD0: + case 0xD1: + case 0xD2: + case 0xD3: + case 0xD4: + case 0xD5: + case 0xD6: + case 0xD7: + case 0xD8: + case 0xD9: + case 0xDA: + case 0xDB: + case 0xDC: + case 0xDD: + case 0xDE: + case 0xDF: + { + if (JSON_UNLIKELY(not next_byte_in_range({0x80, 0xBF}))) + { + return token_type::parse_error; + } + break; + } + + // U+0800..U+0FFF: bytes E0 A0..BF 80..BF + case 0xE0: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0xA0, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+1000..U+CFFF: bytes E1..EC 80..BF 80..BF + // U+E000..U+FFFF: bytes EE..EF 80..BF 80..BF + case 0xE1: + case 0xE2: + case 0xE3: + case 0xE4: + case 0xE5: + case 0xE6: + case 0xE7: + case 0xE8: + case 0xE9: + case 0xEA: + case 0xEB: + case 0xEC: + case 0xEE: + case 0xEF: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+D000..U+D7FF: bytes ED 80..9F 80..BF + case 0xED: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0x9F, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+10000..U+3FFFF F0 90..BF 80..BF 80..BF + case 0xF0: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0x90, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+40000..U+FFFFF F1..F3 80..BF 80..BF 80..BF + case 0xF1: + case 0xF2: + case 0xF3: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+100000..U+10FFFF F4 80..8F 80..BF 80..BF + case 0xF4: + { + if (JSON_UNLIKELY(not (next_byte_in_range({0x80, 0x8F, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // remaining bytes (80..C1 and F5..FF) are ill-formed + default: + { + error_message = "invalid string: ill-formed UTF-8 byte"; + return token_type::parse_error; + } + } + } + } + + static void strtof(float& f, const char* str, char** endptr) noexcept + { + f = std::strtof(str, endptr); + } + + static void strtof(double& f, const char* str, char** endptr) noexcept + { + f = std::strtod(str, endptr); + } + + static void strtof(long double& f, const char* str, char** endptr) noexcept + { + f = std::strtold(str, endptr); + } + + /*! + @brief scan a number literal + + This function scans a string according to Sect. 6 of RFC 7159. + + The function is realized with a deterministic finite state machine derived + from the grammar described in RFC 7159. Starting in state "init", the + input is read and used to determined the next state. Only state "done" + accepts the number. State "error" is a trap state to model errors. In the + table below, "anything" means any character but the ones listed before. + + state | 0 | 1-9 | e E | + | - | . | anything + ---------|----------|----------|----------|---------|---------|----------|----------- + init | zero | any1 | [error] | [error] | minus | [error] | [error] + minus | zero | any1 | [error] | [error] | [error] | [error] | [error] + zero | done | done | exponent | done | done | decimal1 | done + any1 | any1 | any1 | exponent | done | done | decimal1 | done + decimal1 | decimal2 | [error] | [error] | [error] | [error] | [error] | [error] + decimal2 | decimal2 | decimal2 | exponent | done | done | done | done + exponent | any2 | any2 | [error] | sign | sign | [error] | [error] + sign | any2 | any2 | [error] | [error] | [error] | [error] | [error] + any2 | any2 | any2 | done | done | done | done | done + + The state machine is realized with one label per state (prefixed with + "scan_number_") and `goto` statements between them. The state machine + contains cycles, but any cycle can be left when EOF is read. Therefore, + the function is guaranteed to terminate. + + During scanning, the read bytes are stored in token_buffer. This string is + then converted to a signed integer, an unsigned integer, or a + floating-point number. + + @return token_type::value_unsigned, token_type::value_integer, or + token_type::value_float if number could be successfully scanned, + token_type::parse_error otherwise + + @note The scanner is independent of the current locale. Internally, the + locale's decimal point is used instead of `.` to work with the + locale-dependent converters. + */ + token_type scan_number() // lgtm [cpp/use-of-goto] + { + // reset token_buffer to store the number's bytes + reset(); + + // the type of the parsed number; initially set to unsigned; will be + // changed if minus sign, decimal point or exponent is read + token_type number_type = token_type::value_unsigned; + + // state (init): we just found out we need to scan a number + switch (current) + { + case '-': + { + add(current); + goto scan_number_minus; + } + + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + // LCOV_EXCL_START + default: + { + // all other characters are rejected outside scan_number() + assert(false); + } + // LCOV_EXCL_STOP + } + +scan_number_minus: + // state: we just parsed a leading minus sign + number_type = token_type::value_integer; + switch (get()) + { + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + default: + { + error_message = "invalid number; expected digit after '-'"; + return token_type::parse_error; + } + } + +scan_number_zero: + // state: we just parse a zero (maybe with a leading minus sign) + switch (get()) + { + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_any1: + // state: we just parsed a number 0-9 (maybe with a leading minus sign) + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_decimal1: + // state: we just parsed a decimal point + number_type = token_type::value_float; + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + default: + { + error_message = "invalid number; expected digit after '.'"; + return token_type::parse_error; + } + } + +scan_number_decimal2: + // we just parsed at least one number after a decimal point + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_exponent: + // we just parsed an exponent + number_type = token_type::value_float; + switch (get()) + { + case '+': + case '-': + { + add(current); + goto scan_number_sign; + } + + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = + "invalid number; expected '+', '-', or digit after exponent"; + return token_type::parse_error; + } + } + +scan_number_sign: + // we just parsed an exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = "invalid number; expected digit after exponent sign"; + return token_type::parse_error; + } + } + +scan_number_any2: + // we just parsed a number after the exponent or exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + goto scan_number_done; + } + +scan_number_done: + // unget the character after the number (we only read it to know that + // we are done scanning a number) + unget(); + + char* endptr = nullptr; + errno = 0; + + // try to parse integers first and fall back to floats + if (number_type == token_type::value_unsigned) + { + const auto x = std::strtoull(token_buffer.data(), &endptr, 10); + + // we checked the number format before + assert(endptr == token_buffer.data() + token_buffer.size()); + + if (errno == 0) + { + value_unsigned = static_cast(x); + if (value_unsigned == x) + { + return token_type::value_unsigned; + } + } + } + else if (number_type == token_type::value_integer) + { + const auto x = std::strtoll(token_buffer.data(), &endptr, 10); + + // we checked the number format before + assert(endptr == token_buffer.data() + token_buffer.size()); + + if (errno == 0) + { + value_integer = static_cast(x); + if (value_integer == x) + { + return token_type::value_integer; + } + } + } + + // this code is reached if we parse a floating-point number or if an + // integer conversion above failed + strtof(value_float, token_buffer.data(), &endptr); + + // we checked the number format before + assert(endptr == token_buffer.data() + token_buffer.size()); + + return token_type::value_float; + } + + /*! + @param[in] literal_text the literal text to expect + @param[in] length the length of the passed literal text + @param[in] return_type the token type to return on success + */ + token_type scan_literal(const char* literal_text, const std::size_t length, + token_type return_type) + { + assert(current == literal_text[0]); + for (std::size_t i = 1; i < length; ++i) + { + if (JSON_UNLIKELY(get() != literal_text[i])) + { + error_message = "invalid literal"; + return token_type::parse_error; + } + } + return return_type; + } + + ///////////////////// + // input management + ///////////////////// + + /// reset token_buffer; current character is beginning of token + void reset() noexcept + { + token_buffer.clear(); + token_string.clear(); + token_string.push_back(std::char_traits::to_char_type(current)); + } + + /* + @brief get next character from the input + + This function provides the interface to the used input adapter. It does + not throw in case the input reached EOF, but returns a + `std::char_traits::eof()` in that case. Stores the scanned characters + for use in error messages. + + @return character read from the input + */ + std::char_traits::int_type get() + { + ++position.chars_read_total; + ++position.chars_read_current_line; + + if (next_unget) + { + // just reset the next_unget variable and work with current + next_unget = false; + } + else + { + current = ia->get_character(); + } + + if (JSON_LIKELY(current != std::char_traits::eof())) + { + token_string.push_back(std::char_traits::to_char_type(current)); + } + + if (current == '\n') + { + ++position.lines_read; + position.chars_read_current_line = 0; + } + + return current; + } + + /*! + @brief unget current character (read it again on next get) + + We implement unget by setting variable next_unget to true. The input is not + changed - we just simulate ungetting by modifying chars_read_total, + chars_read_current_line, and token_string. The next call to get() will + behave as if the unget character is read again. + */ + void unget() + { + next_unget = true; + + --position.chars_read_total; + + // in case we "unget" a newline, we have to also decrement the lines_read + if (position.chars_read_current_line == 0) + { + if (position.lines_read > 0) + { + --position.lines_read; + } + } + else + { + --position.chars_read_current_line; + } + + if (JSON_LIKELY(current != std::char_traits::eof())) + { + assert(not token_string.empty()); + token_string.pop_back(); + } + } + + /// add a character to token_buffer + void add(int c) + { + token_buffer.push_back(std::char_traits::to_char_type(c)); + } + + public: + ///////////////////// + // value getters + ///////////////////// + + /// return integer value + constexpr number_integer_t get_number_integer() const noexcept + { + return value_integer; + } + + /// return unsigned integer value + constexpr number_unsigned_t get_number_unsigned() const noexcept + { + return value_unsigned; + } + + /// return floating-point value + constexpr number_float_t get_number_float() const noexcept + { + return value_float; + } + + /// return current string value (implicitly resets the token; useful only once) + string_t& get_string() + { + return token_buffer; + } + + ///////////////////// + // diagnostics + ///////////////////// + + /// return position of last read token + constexpr position_t get_position() const noexcept + { + return position; + } + + /// return the last read token (for errors only). Will never contain EOF + /// (an arbitrary value that is not a valid char value, often -1), because + /// 255 may legitimately occur. May contain NUL, which should be escaped. + std::string get_token_string() const + { + // escape control characters + std::string result; + for (const auto c : token_string) + { + if ('\x00' <= c and c <= '\x1F') + { + // escape control characters + std::array cs{}; + (std::snprintf)(cs.data(), cs.size(), "", static_cast(c)); + result += cs.data(); + } + else + { + // add character as is + result.push_back(c); + } + } + + return result; + } + + /// return syntax error message + constexpr const char* get_error_message() const noexcept + { + return error_message; + } + + ///////////////////// + // actual scanner + ///////////////////// + + /*! + @brief skip the UTF-8 byte order mark + @return true iff there is no BOM or the correct BOM has been skipped + */ + bool skip_bom() + { + if (get() == 0xEF) + { + // check if we completely parse the BOM + return get() == 0xBB and get() == 0xBF; + } + + // the first character is not the beginning of the BOM; unget it to + // process is later + unget(); + return true; + } + + token_type scan() + { + // initially, skip the BOM + if (position.chars_read_total == 0 and not skip_bom()) + { + error_message = "invalid BOM; must be 0xEF 0xBB 0xBF if given"; + return token_type::parse_error; + } + + // read next character and ignore whitespace + do + { + get(); + } + while (current == ' ' or current == '\t' or current == '\n' or current == '\r'); + + switch (current) + { + // structural characters + case '[': + return token_type::begin_array; + case ']': + return token_type::end_array; + case '{': + return token_type::begin_object; + case '}': + return token_type::end_object; + case ':': + return token_type::name_separator; + case ',': + return token_type::value_separator; + + // literals + case 't': + return scan_literal("true", 4, token_type::literal_true); + case 'f': + return scan_literal("false", 5, token_type::literal_false); + case 'n': + return scan_literal("null", 4, token_type::literal_null); + + // string + case '\"': + return scan_string(); + + // number + case '-': + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + return scan_number(); + + // end of input (the null byte is needed when parsing from + // string literals) + case '\0': + case std::char_traits::eof(): + return token_type::end_of_input; + + // error + default: + error_message = "invalid literal"; + return token_type::parse_error; + } + } + + private: + /// input adapter + detail::input_adapter_t ia = nullptr; + + /// the current character + std::char_traits::int_type current = std::char_traits::eof(); + + /// whether the next get() call should just return current + bool next_unget = false; + + /// the start position of the current token + position_t position {}; + + /// raw input token string (for error messages) + std::vector token_string {}; + + /// buffer for variable-length tokens (numbers, strings) + string_t token_buffer {}; + + /// a description of occurred lexer errors + const char* error_message = ""; + + // number values + number_integer_t value_integer = 0; + number_unsigned_t value_unsigned = 0; + number_float_t value_float = 0; + + /// the decimal point + const char decimal_point_char = '.'; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // assert +#include // isfinite +#include // uint8_t +#include // function +#include // string +#include // move +#include // vector + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +//////////// +// parser // +//////////// + +/*! +@brief syntax analysis + +This class implements a recursive decent parser. +*/ +template +class parser +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using lexer_t = lexer; + using token_type = typename lexer_t::token_type; + + public: + enum class parse_event_t : uint8_t + { + /// the parser read `{` and started to process a JSON object + object_start, + /// the parser read `}` and finished processing a JSON object + object_end, + /// the parser read `[` and started to process a JSON array + array_start, + /// the parser read `]` and finished processing a JSON array + array_end, + /// the parser read a key of a value in an object + key, + /// the parser finished reading a JSON value + value + }; + + using parser_callback_t = + std::function; + + /// a parser reading from an input adapter + explicit parser(detail::input_adapter_t&& adapter, + const parser_callback_t cb = nullptr, + const bool allow_exceptions_ = true) + : callback(cb), m_lexer(std::move(adapter)), allow_exceptions(allow_exceptions_) + { + // read first token + get_token(); + } + + /*! + @brief public parser interface + + @param[in] strict whether to expect the last token to be EOF + @param[in,out] result parsed JSON value + + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ + void parse(const bool strict, BasicJsonType& result) + { + if (callback) + { + json_sax_dom_callback_parser sdp(result, callback, allow_exceptions); + sax_parse_internal(&sdp); + result.assert_invariant(); + + // in strict mode, input must be completely read + if (strict and (get_token() != token_type::end_of_input)) + { + sdp.parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_of_input, "value"))); + } + + // in case of an error, return discarded value + if (sdp.is_errored()) + { + result = value_t::discarded; + return; + } + + // set top-level value to null if it was discarded by the callback + // function + if (result.is_discarded()) + { + result = nullptr; + } + } + else + { + json_sax_dom_parser sdp(result, allow_exceptions); + sax_parse_internal(&sdp); + result.assert_invariant(); + + // in strict mode, input must be completely read + if (strict and (get_token() != token_type::end_of_input)) + { + sdp.parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_of_input, "value"))); + } + + // in case of an error, return discarded value + if (sdp.is_errored()) + { + result = value_t::discarded; + return; + } + } + } + + /*! + @brief public accept interface + + @param[in] strict whether to expect the last token to be EOF + @return whether the input is a proper JSON text + */ + bool accept(const bool strict = true) + { + json_sax_acceptor sax_acceptor; + return sax_parse(&sax_acceptor, strict); + } + + template + bool sax_parse(SAX* sax, const bool strict = true) + { + (void)detail::is_sax_static_asserts {}; + const bool result = sax_parse_internal(sax); + + // strict mode: next byte must be EOF + if (result and strict and (get_token() != token_type::end_of_input)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_of_input, "value"))); + } + + return result; + } + + private: + template + bool sax_parse_internal(SAX* sax) + { + // stack to remember the hierarchy of structured values we are parsing + // true = array; false = object + std::vector states; + // value to avoid a goto (see comment where set to true) + bool skip_to_state_evaluation = false; + + while (true) + { + if (not skip_to_state_evaluation) + { + // invariant: get_token() was called before each iteration + switch (last_token) + { + case token_type::begin_object: + { + if (JSON_UNLIKELY(not sax->start_object(std::size_t(-1)))) + { + return false; + } + + // closing } -> we are done + if (get_token() == token_type::end_object) + { + if (JSON_UNLIKELY(not sax->end_object())) + { + return false; + } + break; + } + + // parse key + if (JSON_UNLIKELY(last_token != token_type::value_string)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::value_string, "object key"))); + } + if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) + { + return false; + } + + // parse separator (:) + if (JSON_UNLIKELY(get_token() != token_type::name_separator)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::name_separator, "object separator"))); + } + + // remember we are now inside an object + states.push_back(false); + + // parse values + get_token(); + continue; + } + + case token_type::begin_array: + { + if (JSON_UNLIKELY(not sax->start_array(std::size_t(-1)))) + { + return false; + } + + // closing ] -> we are done + if (get_token() == token_type::end_array) + { + if (JSON_UNLIKELY(not sax->end_array())) + { + return false; + } + break; + } + + // remember we are now inside an array + states.push_back(true); + + // parse values (no need to call get_token) + continue; + } + + case token_type::value_float: + { + const auto res = m_lexer.get_number_float(); + + if (JSON_UNLIKELY(not std::isfinite(res))) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); + } + else + { + if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) + { + return false; + } + break; + } + } + + case token_type::literal_false: + { + if (JSON_UNLIKELY(not sax->boolean(false))) + { + return false; + } + break; + } + + case token_type::literal_null: + { + if (JSON_UNLIKELY(not sax->null())) + { + return false; + } + break; + } + + case token_type::literal_true: + { + if (JSON_UNLIKELY(not sax->boolean(true))) + { + return false; + } + break; + } + + case token_type::value_integer: + { + if (JSON_UNLIKELY(not sax->number_integer(m_lexer.get_number_integer()))) + { + return false; + } + break; + } + + case token_type::value_string: + { + if (JSON_UNLIKELY(not sax->string(m_lexer.get_string()))) + { + return false; + } + break; + } + + case token_type::value_unsigned: + { + if (JSON_UNLIKELY(not sax->number_unsigned(m_lexer.get_number_unsigned()))) + { + return false; + } + break; + } + + case token_type::parse_error: + { + // using "uninitialized" to avoid "expected" message + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::uninitialized, "value"))); + } + + default: // the last token was unexpected + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::literal_or_value, "value"))); + } + } + } + else + { + skip_to_state_evaluation = false; + } + + // we reached this line after we successfully parsed a value + if (states.empty()) + { + // empty stack: we reached the end of the hierarchy: done + return true; + } + else + { + if (states.back()) // array + { + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse a new value + get_token(); + continue; + } + + // closing ] + if (JSON_LIKELY(last_token == token_type::end_array)) + { + if (JSON_UNLIKELY(not sax->end_array())) + { + return false; + } + + // We are done with this array. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + else + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_array, "array"))); + } + } + else // object + { + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse key + if (JSON_UNLIKELY(get_token() != token_type::value_string)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::value_string, "object key"))); + } + else + { + if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) + { + return false; + } + } + + // parse separator (:) + if (JSON_UNLIKELY(get_token() != token_type::name_separator)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::name_separator, "object separator"))); + } + + // parse values + get_token(); + continue; + } + + // closing } + if (JSON_LIKELY(last_token == token_type::end_object)) + { + if (JSON_UNLIKELY(not sax->end_object())) + { + return false; + } + + // We are done with this object. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + else + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_object, "object"))); + } + } + } + } + } + + /// get next token from lexer + token_type get_token() + { + return (last_token = m_lexer.scan()); + } + + std::string exception_message(const token_type expected, const std::string& context) + { + std::string error_msg = "syntax error "; + + if (not context.empty()) + { + error_msg += "while parsing " + context + " "; + } + + error_msg += "- "; + + if (last_token == token_type::parse_error) + { + error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + + m_lexer.get_token_string() + "'"; + } + else + { + error_msg += "unexpected " + std::string(lexer_t::token_type_name(last_token)); + } + + if (expected != token_type::uninitialized) + { + error_msg += "; expected " + std::string(lexer_t::token_type_name(expected)); + } + + return error_msg; + } + + private: + /// callback function + const parser_callback_t callback = nullptr; + /// the type of the last read token + token_type last_token = token_type::uninitialized; + /// the lexer + lexer_t m_lexer; + /// whether to throw exceptions in case of errors + const bool allow_exceptions = true; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +// #include + + +#include // ptrdiff_t +#include // numeric_limits + +namespace nlohmann +{ +namespace detail +{ +/* +@brief an iterator for primitive JSON types + +This class models an iterator for primitive JSON types (boolean, number, +string). It's only purpose is to allow the iterator/const_iterator classes +to "iterate" over primitive values. Internally, the iterator is modeled by +a `difference_type` variable. Value begin_value (`0`) models the begin, +end_value (`1`) models past the end. +*/ +class primitive_iterator_t +{ + private: + using difference_type = std::ptrdiff_t; + static constexpr difference_type begin_value = 0; + static constexpr difference_type end_value = begin_value + 1; + + /// iterator as signed integer type + difference_type m_it = (std::numeric_limits::min)(); + + public: + constexpr difference_type get_value() const noexcept + { + return m_it; + } + + /// set iterator to a defined beginning + void set_begin() noexcept + { + m_it = begin_value; + } + + /// set iterator to a defined past the end + void set_end() noexcept + { + m_it = end_value; + } + + /// return whether the iterator can be dereferenced + constexpr bool is_begin() const noexcept + { + return m_it == begin_value; + } + + /// return whether the iterator is at end + constexpr bool is_end() const noexcept + { + return m_it == end_value; + } + + friend constexpr bool operator==(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it == rhs.m_it; + } + + friend constexpr bool operator<(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it < rhs.m_it; + } + + primitive_iterator_t operator+(difference_type n) noexcept + { + auto result = *this; + result += n; + return result; + } + + friend constexpr difference_type operator-(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it - rhs.m_it; + } + + primitive_iterator_t& operator++() noexcept + { + ++m_it; + return *this; + } + + primitive_iterator_t const operator++(int) noexcept + { + auto result = *this; + ++m_it; + return result; + } + + primitive_iterator_t& operator--() noexcept + { + --m_it; + return *this; + } + + primitive_iterator_t const operator--(int) noexcept + { + auto result = *this; + --m_it; + return result; + } + + primitive_iterator_t& operator+=(difference_type n) noexcept + { + m_it += n; + return *this; + } + + primitive_iterator_t& operator-=(difference_type n) noexcept + { + m_it -= n; + return *this; + } +}; +} // namespace detail +} // namespace nlohmann + + +namespace nlohmann +{ +namespace detail +{ +/*! +@brief an iterator value + +@note This structure could easily be a union, but MSVC currently does not allow +unions members with complex constructors, see https://github.com/nlohmann/json/pull/105. +*/ +template struct internal_iterator +{ + /// iterator for JSON objects + typename BasicJsonType::object_t::iterator object_iterator {}; + /// iterator for JSON arrays + typename BasicJsonType::array_t::iterator array_iterator {}; + /// generic iterator for all other types + primitive_iterator_t primitive_iterator {}; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // not +#include // iterator, random_access_iterator_tag, bidirectional_iterator_tag, advance, next +#include // conditional, is_const, remove_const + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +// forward declare, to be able to friend it later on +template class iteration_proxy; +template class iteration_proxy_value; + +/*! +@brief a template for a bidirectional iterator for the @ref basic_json class +This class implements a both iterators (iterator and const_iterator) for the +@ref basic_json class. +@note An iterator is called *initialized* when a pointer to a JSON value has + been set (e.g., by a constructor or a copy assignment). If the iterator is + default-constructed, it is *uninitialized* and most methods are undefined. + **The library uses assertions to detect calls on uninitialized iterators.** +@requirement The class satisfies the following concept requirements: +- +[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): + The iterator that can be moved can be moved in both directions (i.e. + incremented and decremented). +@since version 1.0.0, simplified in version 2.0.9, change to bidirectional + iterators in version 3.0.0 (see https://github.com/nlohmann/json/issues/593) +*/ +template +class iter_impl +{ + /// allow basic_json to access private members + friend iter_impl::value, typename std::remove_const::type, const BasicJsonType>::type>; + friend BasicJsonType; + friend iteration_proxy; + friend iteration_proxy_value; + + using object_t = typename BasicJsonType::object_t; + using array_t = typename BasicJsonType::array_t; + // make sure BasicJsonType is basic_json or const basic_json + static_assert(is_basic_json::type>::value, + "iter_impl only accepts (const) basic_json"); + + public: + + /// The std::iterator class template (used as a base class to provide typedefs) is deprecated in C++17. + /// The C++ Standard has never required user-defined iterators to derive from std::iterator. + /// A user-defined iterator should provide publicly accessible typedefs named + /// iterator_category, value_type, difference_type, pointer, and reference. + /// Note that value_type is required to be non-const, even for constant iterators. + using iterator_category = std::bidirectional_iterator_tag; + + /// the type of the values when the iterator is dereferenced + using value_type = typename BasicJsonType::value_type; + /// a type to represent differences between iterators + using difference_type = typename BasicJsonType::difference_type; + /// defines a pointer to the type iterated over (value_type) + using pointer = typename std::conditional::value, + typename BasicJsonType::const_pointer, + typename BasicJsonType::pointer>::type; + /// defines a reference to the type iterated over (value_type) + using reference = + typename std::conditional::value, + typename BasicJsonType::const_reference, + typename BasicJsonType::reference>::type; + + /// default constructor + iter_impl() = default; + + /*! + @brief constructor for a given JSON instance + @param[in] object pointer to a JSON object for this iterator + @pre object != nullptr + @post The iterator is initialized; i.e. `m_object != nullptr`. + */ + explicit iter_impl(pointer object) noexcept : m_object(object) + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = typename object_t::iterator(); + break; + } + + case value_t::array: + { + m_it.array_iterator = typename array_t::iterator(); + break; + } + + default: + { + m_it.primitive_iterator = primitive_iterator_t(); + break; + } + } + } + + /*! + @note The conventional copy constructor and copy assignment are implicitly + defined. Combined with the following converting constructor and + assignment, they support: (1) copy from iterator to iterator, (2) + copy from const iterator to const iterator, and (3) conversion from + iterator to const iterator. However conversion from const iterator + to iterator is not defined. + */ + + /*! + @brief converting constructor + @param[in] other non-const iterator to copy from + @note It is not checked whether @a other is initialized. + */ + iter_impl(const iter_impl::type>& other) noexcept + : m_object(other.m_object), m_it(other.m_it) {} + + /*! + @brief converting assignment + @param[in,out] other non-const iterator to copy from + @return const/non-const iterator + @note It is not checked whether @a other is initialized. + */ + iter_impl& operator=(const iter_impl::type>& other) noexcept + { + m_object = other.m_object; + m_it = other.m_it; + return *this; + } + + private: + /*! + @brief set the iterator to the first value + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + void set_begin() noexcept + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = m_object->m_value.object->begin(); + break; + } + + case value_t::array: + { + m_it.array_iterator = m_object->m_value.array->begin(); + break; + } + + case value_t::null: + { + // set to end so begin()==end() is true: null is empty + m_it.primitive_iterator.set_end(); + break; + } + + default: + { + m_it.primitive_iterator.set_begin(); + break; + } + } + } + + /*! + @brief set the iterator past the last value + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + void set_end() noexcept + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = m_object->m_value.object->end(); + break; + } + + case value_t::array: + { + m_it.array_iterator = m_object->m_value.array->end(); + break; + } + + default: + { + m_it.primitive_iterator.set_end(); + break; + } + } + } + + public: + /*! + @brief return a reference to the value pointed to by the iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference operator*() const + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + assert(m_it.object_iterator != m_object->m_value.object->end()); + return m_it.object_iterator->second; + } + + case value_t::array: + { + assert(m_it.array_iterator != m_object->m_value.array->end()); + return *m_it.array_iterator; + } + + case value_t::null: + JSON_THROW(invalid_iterator::create(214, "cannot get value")); + + default: + { + if (JSON_LIKELY(m_it.primitive_iterator.is_begin())) + { + return *m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value")); + } + } + } + + /*! + @brief dereference the iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + pointer operator->() const + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + assert(m_it.object_iterator != m_object->m_value.object->end()); + return &(m_it.object_iterator->second); + } + + case value_t::array: + { + assert(m_it.array_iterator != m_object->m_value.array->end()); + return &*m_it.array_iterator; + } + + default: + { + if (JSON_LIKELY(m_it.primitive_iterator.is_begin())) + { + return m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value")); + } + } + } + + /*! + @brief post-increment (it++) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl const operator++(int) + { + auto result = *this; + ++(*this); + return result; + } + + /*! + @brief pre-increment (++it) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator++() + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + std::advance(m_it.object_iterator, 1); + break; + } + + case value_t::array: + { + std::advance(m_it.array_iterator, 1); + break; + } + + default: + { + ++m_it.primitive_iterator; + break; + } + } + + return *this; + } + + /*! + @brief post-decrement (it--) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl const operator--(int) + { + auto result = *this; + --(*this); + return result; + } + + /*! + @brief pre-decrement (--it) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator--() + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + std::advance(m_it.object_iterator, -1); + break; + } + + case value_t::array: + { + std::advance(m_it.array_iterator, -1); + break; + } + + default: + { + --m_it.primitive_iterator; + break; + } + } + + return *this; + } + + /*! + @brief comparison: equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator==(const iter_impl& other) const + { + // if objects are not the same, the comparison is undefined + if (JSON_UNLIKELY(m_object != other.m_object)) + { + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); + } + + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + return (m_it.object_iterator == other.m_it.object_iterator); + + case value_t::array: + return (m_it.array_iterator == other.m_it.array_iterator); + + default: + return (m_it.primitive_iterator == other.m_it.primitive_iterator); + } + } + + /*! + @brief comparison: not equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator!=(const iter_impl& other) const + { + return not operator==(other); + } + + /*! + @brief comparison: smaller + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator<(const iter_impl& other) const + { + // if objects are not the same, the comparison is undefined + if (JSON_UNLIKELY(m_object != other.m_object)) + { + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers")); + } + + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(213, "cannot compare order of object iterators")); + + case value_t::array: + return (m_it.array_iterator < other.m_it.array_iterator); + + default: + return (m_it.primitive_iterator < other.m_it.primitive_iterator); + } + } + + /*! + @brief comparison: less than or equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator<=(const iter_impl& other) const + { + return not other.operator < (*this); + } + + /*! + @brief comparison: greater than + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator>(const iter_impl& other) const + { + return not operator<=(other); + } + + /*! + @brief comparison: greater than or equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator>=(const iter_impl& other) const + { + return not operator<(other); + } + + /*! + @brief add to iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator+=(difference_type i) + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); + + case value_t::array: + { + std::advance(m_it.array_iterator, i); + break; + } + + default: + { + m_it.primitive_iterator += i; + break; + } + } + + return *this; + } + + /*! + @brief subtract from iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator-=(difference_type i) + { + return operator+=(-i); + } + + /*! + @brief add to iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl operator+(difference_type i) const + { + auto result = *this; + result += i; + return result; + } + + /*! + @brief addition of distance and iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + friend iter_impl operator+(difference_type i, const iter_impl& it) + { + auto result = it; + result += i; + return result; + } + + /*! + @brief subtract from iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl operator-(difference_type i) const + { + auto result = *this; + result -= i; + return result; + } + + /*! + @brief return difference + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + difference_type operator-(const iter_impl& other) const + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators")); + + case value_t::array: + return m_it.array_iterator - other.m_it.array_iterator; + + default: + return m_it.primitive_iterator - other.m_it.primitive_iterator; + } + } + + /*! + @brief access to successor + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference operator[](difference_type n) const + { + assert(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(208, "cannot use operator[] for object iterators")); + + case value_t::array: + return *std::next(m_it.array_iterator, n); + + case value_t::null: + JSON_THROW(invalid_iterator::create(214, "cannot get value")); + + default: + { + if (JSON_LIKELY(m_it.primitive_iterator.get_value() == -n)) + { + return *m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value")); + } + } + } + + /*! + @brief return the key of an object iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + const typename object_t::key_type& key() const + { + assert(m_object != nullptr); + + if (JSON_LIKELY(m_object->is_object())) + { + return m_it.object_iterator->first; + } + + JSON_THROW(invalid_iterator::create(207, "cannot use key() for non-object iterators")); + } + + /*! + @brief return the value of an iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference value() const + { + return operator*(); + } + + private: + /// associated JSON instance + pointer m_object = nullptr; + /// the actual iterator of the associated instance + internal_iterator::type> m_it {}; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // ptrdiff_t +#include // reverse_iterator +#include // declval + +namespace nlohmann +{ +namespace detail +{ +////////////////////// +// reverse_iterator // +////////////////////// + +/*! +@brief a template for a reverse iterator class + +@tparam Base the base iterator type to reverse. Valid types are @ref +iterator (to create @ref reverse_iterator) and @ref const_iterator (to +create @ref const_reverse_iterator). + +@requirement The class satisfies the following concept requirements: +- +[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): + The iterator that can be moved can be moved in both directions (i.e. + incremented and decremented). +- [OutputIterator](https://en.cppreference.com/w/cpp/named_req/OutputIterator): + It is possible to write to the pointed-to element (only if @a Base is + @ref iterator). + +@since version 1.0.0 +*/ +template +class json_reverse_iterator : public std::reverse_iterator +{ + public: + using difference_type = std::ptrdiff_t; + /// shortcut to the reverse iterator adapter + using base_iterator = std::reverse_iterator; + /// the reference type for the pointed-to element + using reference = typename Base::reference; + + /// create reverse iterator from iterator + explicit json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept + : base_iterator(it) {} + + /// create reverse iterator from base class + explicit json_reverse_iterator(const base_iterator& it) noexcept : base_iterator(it) {} + + /// post-increment (it++) + json_reverse_iterator const operator++(int) + { + return static_cast(base_iterator::operator++(1)); + } + + /// pre-increment (++it) + json_reverse_iterator& operator++() + { + return static_cast(base_iterator::operator++()); + } + + /// post-decrement (it--) + json_reverse_iterator const operator--(int) + { + return static_cast(base_iterator::operator--(1)); + } + + /// pre-decrement (--it) + json_reverse_iterator& operator--() + { + return static_cast(base_iterator::operator--()); + } + + /// add to iterator + json_reverse_iterator& operator+=(difference_type i) + { + return static_cast(base_iterator::operator+=(i)); + } + + /// add to iterator + json_reverse_iterator operator+(difference_type i) const + { + return static_cast(base_iterator::operator+(i)); + } + + /// subtract from iterator + json_reverse_iterator operator-(difference_type i) const + { + return static_cast(base_iterator::operator-(i)); + } + + /// return difference + difference_type operator-(const json_reverse_iterator& other) const + { + return base_iterator(*this) - base_iterator(other); + } + + /// access to successor + reference operator[](difference_type n) const + { + return *(this->operator+(n)); + } + + /// return the key of an object iterator + auto key() const -> decltype(std::declval().key()) + { + auto it = --this->base(); + return it.key(); + } + + /// return the value of an iterator + reference value() const + { + auto it = --this->base(); + return it.operator * (); + } +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // all_of +#include // assert +#include // accumulate +#include // string +#include // move +#include // vector + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +template +class json_pointer +{ + // allow basic_json to access private members + NLOHMANN_BASIC_JSON_TPL_DECLARATION + friend class basic_json; + + public: + /*! + @brief create JSON pointer + + Create a JSON pointer according to the syntax described in + [Section 3 of RFC6901](https://tools.ietf.org/html/rfc6901#section-3). + + @param[in] s string representing the JSON pointer; if omitted, the empty + string is assumed which references the whole JSON value + + @throw parse_error.107 if the given JSON pointer @a s is nonempty and does + not begin with a slash (`/`); see example below + + @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s is + not followed by `0` (representing `~`) or `1` (representing `/`); see + example below + + @liveexample{The example shows the construction several valid JSON pointers + as well as the exceptional behavior.,json_pointer} + + @since version 2.0.0 + */ + explicit json_pointer(const std::string& s = "") + : reference_tokens(split(s)) + {} + + /*! + @brief return a string representation of the JSON pointer + + @invariant For each JSON pointer `ptr`, it holds: + @code {.cpp} + ptr == json_pointer(ptr.to_string()); + @endcode + + @return a string representation of the JSON pointer + + @liveexample{The example shows the result of `to_string`., + json_pointer__to_string} + + @since version 2.0.0 + */ + std::string to_string() const + { + return std::accumulate(reference_tokens.begin(), reference_tokens.end(), + std::string{}, + [](const std::string & a, const std::string & b) + { + return a + "/" + escape(b); + }); + } + + /// @copydoc to_string() + operator std::string() const + { + return to_string(); + } + + /*! + @brief append another JSON pointer at the end of this JSON pointer + */ + json_pointer& operator/=(const json_pointer& ptr) + { + reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); + return *this; + } + + /// @copydoc push_back(std::string&&) + json_pointer& operator/=(std::string token) + { + push_back(std::move(token)); + return *this; + } + + /// @copydoc operator/=(std::string) + json_pointer& operator/=(std::size_t array_index) + { + return *this /= std::to_string(array_index); + } + + /*! + @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + */ + friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) + { + return json_pointer(left_ptr) /= right_ptr; + } + + /*! + @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& ptr, std::string token) + { + return json_pointer(ptr) /= std::move(token); + } + + /*! + @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + */ + friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) + { + return json_pointer(lhs) /= array_index; + } + + /*! + @brief create a new JSON pointer that is the parent of this JSON pointer + */ + json_pointer parent_pointer() const + { + if (empty()) + { + return *this; + } + + json_pointer res = *this; + res.pop_back(); + return res; + } + + /*! + @param[in] s reference token to be converted into an array index + + @return integer representation of @a s + + @throw out_of_range.404 if string @a s could not be converted to an integer + */ + static int array_index(const std::string& s) + { + std::size_t processed_chars = 0; + const int res = std::stoi(s, &processed_chars); + + // check if the string was completely read + if (JSON_UNLIKELY(processed_chars != s.size())) + { + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); + } + + return res; + } + + /*! + @brief remove and return last reference token + @throw out_of_range.405 if JSON pointer has no parent + */ + std::string pop_back() + { + if (JSON_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); + } + + auto last = reference_tokens.back(); + reference_tokens.pop_back(); + return last; + } + + /*! + @brief append an unescaped token at the end of the reference pointer + */ + void push_back(const std::string& token) + { + reference_tokens.push_back(token); + } + + /// @copydoc push_back(const std::string&) + void push_back(std::string&& token) + { + reference_tokens.push_back(std::move(token)); + } + + /// return whether pointer points to the root document + bool empty() const noexcept + { + return reference_tokens.empty(); + } + + private: + json_pointer top() const + { + if (JSON_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); + } + + json_pointer result = *this; + result.reference_tokens = {reference_tokens[0]}; + return result; + } + + /*! + @brief create and return a reference to the pointed to value + + @complexity Linear in the number of reference tokens. + + @throw parse_error.109 if array index is not a number + @throw type_error.313 if value cannot be unflattened + */ + BasicJsonType& get_and_create(BasicJsonType& j) const + { + using size_type = typename BasicJsonType::size_type; + auto result = &j; + + // in case no reference tokens exist, return a reference to the JSON value + // j which will be overwritten by a primitive value + for (const auto& reference_token : reference_tokens) + { + switch (result->m_type) + { + case detail::value_t::null: + { + if (reference_token == "0") + { + // start a new array if reference token is 0 + result = &result->operator[](0); + } + else + { + // start a new object otherwise + result = &result->operator[](reference_token); + } + break; + } + + case detail::value_t::object: + { + // create an entry in the object + result = &result->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + // create an entry in the array + JSON_TRY + { + result = &result->operator[](static_cast(array_index(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); + } + break; + } + + /* + The following code is only reached if there exists a reference + token _and_ the current value is primitive. In this case, we have + an error situation, because primitive values may only occur as + single value; that is, with an empty list of reference tokens. + */ + default: + JSON_THROW(detail::type_error::create(313, "invalid value to unflatten")); + } + } + + return *result; + } + + /*! + @brief return a reference to the pointed to value + + @note This version does not throw if a value is not present, but tries to + create nested values instead. For instance, calling this function + with pointer `"/this/that"` on a null value is equivalent to calling + `operator[]("this").operator[]("that")` on that value, effectively + changing the null value to an object. + + @param[in] ptr a JSON value + + @return reference to the JSON value pointed to by the JSON pointer + + @complexity Linear in the length of the JSON pointer. + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + BasicJsonType& get_unchecked(BasicJsonType* ptr) const + { + using size_type = typename BasicJsonType::size_type; + for (const auto& reference_token : reference_tokens) + { + // convert null values to arrays or objects before continuing + if (ptr->m_type == detail::value_t::null) + { + // check if reference token is a number + const bool nums = + std::all_of(reference_token.begin(), reference_token.end(), + [](const char x) + { + return (x >= '0' and x <= '9'); + }); + + // change value to array for numbers or "-" or to object otherwise + *ptr = (nums or reference_token == "-") + ? detail::value_t::array + : detail::value_t::object; + } + + switch (ptr->m_type) + { + case detail::value_t::object: + { + // use unchecked object access + ptr = &ptr->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) + { + JSON_THROW(detail::parse_error::create(106, 0, + "array index '" + reference_token + + "' must not begin with '0'")); + } + + if (reference_token == "-") + { + // explicitly treat "-" as index beyond the end + ptr = &ptr->operator[](ptr->m_value.array->size()); + } + else + { + // convert array index to number; unchecked access + JSON_TRY + { + ptr = &ptr->operator[]( + static_cast(array_index(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); + } + } + break; + } + + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); + } + } + + return *ptr; + } + + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + BasicJsonType& get_checked(BasicJsonType* ptr) const + { + using size_type = typename BasicJsonType::size_type; + for (const auto& reference_token : reference_tokens) + { + switch (ptr->m_type) + { + case detail::value_t::object: + { + // note: at performs range check + ptr = &ptr->at(reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_UNLIKELY(reference_token == "-")) + { + // "-" always fails the range check + JSON_THROW(detail::out_of_range::create(402, + "array index '-' (" + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); + } + + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) + { + JSON_THROW(detail::parse_error::create(106, 0, + "array index '" + reference_token + + "' must not begin with '0'")); + } + + // note: at performs range check + JSON_TRY + { + ptr = &ptr->at(static_cast(array_index(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); + } + break; + } + + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); + } + } + + return *ptr; + } + + /*! + @brief return a const reference to the pointed to value + + @param[in] ptr a JSON value + + @return const reference to the JSON value pointed to by the JSON + pointer + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + const BasicJsonType& get_unchecked(const BasicJsonType* ptr) const + { + using size_type = typename BasicJsonType::size_type; + for (const auto& reference_token : reference_tokens) + { + switch (ptr->m_type) + { + case detail::value_t::object: + { + // use unchecked object access + ptr = &ptr->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_UNLIKELY(reference_token == "-")) + { + // "-" cannot be used for const access + JSON_THROW(detail::out_of_range::create(402, + "array index '-' (" + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); + } + + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) + { + JSON_THROW(detail::parse_error::create(106, 0, + "array index '" + reference_token + + "' must not begin with '0'")); + } + + // use unchecked array access + JSON_TRY + { + ptr = &ptr->operator[]( + static_cast(array_index(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); + } + break; + } + + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); + } + } + + return *ptr; + } + + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + const BasicJsonType& get_checked(const BasicJsonType* ptr) const + { + using size_type = typename BasicJsonType::size_type; + for (const auto& reference_token : reference_tokens) + { + switch (ptr->m_type) + { + case detail::value_t::object: + { + // note: at performs range check + ptr = &ptr->at(reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_UNLIKELY(reference_token == "-")) + { + // "-" always fails the range check + JSON_THROW(detail::out_of_range::create(402, + "array index '-' (" + std::to_string(ptr->m_value.array->size()) + + ") is out of range")); + } + + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) + { + JSON_THROW(detail::parse_error::create(106, 0, + "array index '" + reference_token + + "' must not begin with '0'")); + } + + // note: at performs range check + JSON_TRY + { + ptr = &ptr->at(static_cast(array_index(reference_token))); + } + JSON_CATCH(std::invalid_argument&) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); + } + break; + } + + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); + } + } + + return *ptr; + } + + /*! + @brief split the string input to reference tokens + + @note This function is only called by the json_pointer constructor. + All exceptions below are documented there. + + @throw parse_error.107 if the pointer is not empty or begins with '/' + @throw parse_error.108 if character '~' is not followed by '0' or '1' + */ + static std::vector split(const std::string& reference_string) + { + std::vector result; + + // special case: empty reference string -> no reference tokens + if (reference_string.empty()) + { + return result; + } + + // check if nonempty reference string begins with slash + if (JSON_UNLIKELY(reference_string[0] != '/')) + { + JSON_THROW(detail::parse_error::create(107, 1, + "JSON pointer must be empty or begin with '/' - was: '" + + reference_string + "'")); + } + + // extract the reference tokens: + // - slash: position of the last read slash (or end of string) + // - start: position after the previous slash + for ( + // search for the first slash after the first character + std::size_t slash = reference_string.find_first_of('/', 1), + // set the beginning of the first reference token + start = 1; + // we can stop if start == 0 (if slash == std::string::npos) + start != 0; + // set the beginning of the next reference token + // (will eventually be 0 if slash == std::string::npos) + start = (slash == std::string::npos) ? 0 : slash + 1, + // find next slash + slash = reference_string.find_first_of('/', start)) + { + // use the text between the beginning of the reference token + // (start) and the last slash (slash). + auto reference_token = reference_string.substr(start, slash - start); + + // check reference tokens are properly escaped + for (std::size_t pos = reference_token.find_first_of('~'); + pos != std::string::npos; + pos = reference_token.find_first_of('~', pos + 1)) + { + assert(reference_token[pos] == '~'); + + // ~ must be followed by 0 or 1 + if (JSON_UNLIKELY(pos == reference_token.size() - 1 or + (reference_token[pos + 1] != '0' and + reference_token[pos + 1] != '1'))) + { + JSON_THROW(detail::parse_error::create(108, 0, "escape character '~' must be followed with '0' or '1'")); + } + } + + // finally, store the reference token + unescape(reference_token); + result.push_back(reference_token); + } + + return result; + } + + /*! + @brief replace all occurrences of a substring by another string + + @param[in,out] s the string to manipulate; changed so that all + occurrences of @a f are replaced with @a t + @param[in] f the substring to replace with @a t + @param[in] t the string to replace @a f + + @pre The search string @a f must not be empty. **This precondition is + enforced with an assertion.** + + @since version 2.0.0 + */ + static void replace_substring(std::string& s, const std::string& f, + const std::string& t) + { + assert(not f.empty()); + for (auto pos = s.find(f); // find first occurrence of f + pos != std::string::npos; // make sure f was found + s.replace(pos, f.size(), t), // replace with t, and + pos = s.find(f, pos + t.size())) // find next occurrence of f + {} + } + + /// escape "~" to "~0" and "/" to "~1" + static std::string escape(std::string s) + { + replace_substring(s, "~", "~0"); + replace_substring(s, "/", "~1"); + return s; + } + + /// unescape "~1" to tilde and "~0" to slash (order is important!) + static void unescape(std::string& s) + { + replace_substring(s, "~1", "/"); + replace_substring(s, "~0", "~"); + } + + /*! + @param[in] reference_string the reference string to the current value + @param[in] value the value to consider + @param[in,out] result the result object to insert values to + + @note Empty objects or arrays are flattened to `null`. + */ + static void flatten(const std::string& reference_string, + const BasicJsonType& value, + BasicJsonType& result) + { + switch (value.m_type) + { + case detail::value_t::array: + { + if (value.m_value.array->empty()) + { + // flatten empty array as null + result[reference_string] = nullptr; + } + else + { + // iterate array and use index as reference string + for (std::size_t i = 0; i < value.m_value.array->size(); ++i) + { + flatten(reference_string + "/" + std::to_string(i), + value.m_value.array->operator[](i), result); + } + } + break; + } + + case detail::value_t::object: + { + if (value.m_value.object->empty()) + { + // flatten empty object as null + result[reference_string] = nullptr; + } + else + { + // iterate object and use keys as reference string + for (const auto& element : *value.m_value.object) + { + flatten(reference_string + "/" + escape(element.first), element.second, result); + } + } + break; + } + + default: + { + // add primitive value with its reference string + result[reference_string] = value; + break; + } + } + } + + /*! + @param[in] value flattened JSON + + @return unflattened JSON + + @throw parse_error.109 if array index is not a number + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitive + @throw type_error.313 if value cannot be unflattened + */ + static BasicJsonType + unflatten(const BasicJsonType& value) + { + if (JSON_UNLIKELY(not value.is_object())) + { + JSON_THROW(detail::type_error::create(314, "only objects can be unflattened")); + } + + BasicJsonType result; + + // iterate the JSON object values + for (const auto& element : *value.m_value.object) + { + if (JSON_UNLIKELY(not element.second.is_primitive())) + { + JSON_THROW(detail::type_error::create(315, "values in object must be primitive")); + } + + // assign value to reference pointed to by JSON pointer; Note that if + // the JSON pointer is "" (i.e., points to the whole value), function + // get_and_create returns a reference to result itself. An assignment + // will then create a primitive value. + json_pointer(element.first).get_and_create(result) = element.second; + } + + return result; + } + + friend bool operator==(json_pointer const& lhs, + json_pointer const& rhs) noexcept + { + return (lhs.reference_tokens == rhs.reference_tokens); + } + + friend bool operator!=(json_pointer const& lhs, + json_pointer const& rhs) noexcept + { + return not (lhs == rhs); + } + + /// the reference tokens + std::vector reference_tokens; +}; +} // namespace nlohmann + +// #include + + +#include +#include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template +class json_ref +{ + public: + using value_type = BasicJsonType; + + json_ref(value_type&& value) + : owned_value(std::move(value)), value_ref(&owned_value), is_rvalue(true) + {} + + json_ref(const value_type& value) + : value_ref(const_cast(&value)), is_rvalue(false) + {} + + json_ref(std::initializer_list init) + : owned_value(init), value_ref(&owned_value), is_rvalue(true) + {} + + template < + class... Args, + enable_if_t::value, int> = 0 > + json_ref(Args && ... args) + : owned_value(std::forward(args)...), value_ref(&owned_value), + is_rvalue(true) {} + + // class should be movable only + json_ref(json_ref&&) noexcept = default; + json_ref(const json_ref&) = delete; + json_ref& operator=(const json_ref&) = delete; + json_ref& operator=(json_ref&&) = delete; + ~json_ref() = default; + + value_type moved_or_copied() const + { + if (is_rvalue) + { + return std::move(*value_ref); + } + return *value_ref; + } + + value_type const& operator*() const + { + return *static_cast(value_ref); + } + + value_type const* operator->() const + { + return static_cast(value_ref); + } + + private: + mutable value_type owned_value = nullptr; + value_type* value_ref = nullptr; + const bool is_rvalue; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + +// #include + // #include @@ -8502,6 +9312,125 @@ class binary_reader // #include +#include // copy +#include // size_t +#include // streamsize +#include // back_inserter +#include // shared_ptr, make_shared +#include // basic_ostream +#include // basic_string +#include // vector + +namespace nlohmann +{ +namespace detail +{ +/// abstract output adapter interface +template struct output_adapter_protocol +{ + virtual void write_character(CharType c) = 0; + virtual void write_characters(const CharType* s, std::size_t length) = 0; + virtual ~output_adapter_protocol() = default; +}; + +/// a type to simplify interfaces +template +using output_adapter_t = std::shared_ptr>; + +/// output adapter for byte vectors +template +class output_vector_adapter : public output_adapter_protocol +{ + public: + explicit output_vector_adapter(std::vector& vec) noexcept + : v(vec) + {} + + void write_character(CharType c) override + { + v.push_back(c); + } + + void write_characters(const CharType* s, std::size_t length) override + { + std::copy(s, s + length, std::back_inserter(v)); + } + + private: + std::vector& v; +}; + +/// output adapter for output streams +template +class output_stream_adapter : public output_adapter_protocol +{ + public: + explicit output_stream_adapter(std::basic_ostream& s) noexcept + : stream(s) + {} + + void write_character(CharType c) override + { + stream.put(c); + } + + void write_characters(const CharType* s, std::size_t length) override + { + stream.write(s, static_cast(length)); + } + + private: + std::basic_ostream& stream; +}; + +/// output adapter for basic_string +template> +class output_string_adapter : public output_adapter_protocol +{ + public: + explicit output_string_adapter(StringType& s) noexcept + : str(s) + {} + + void write_character(CharType c) override + { + str.push_back(c); + } + + void write_characters(const CharType* s, std::size_t length) override + { + str.append(s, length); + } + + private: + StringType& str; +}; + +template> +class output_adapter +{ + public: + output_adapter(std::vector& vec) + : oa(std::make_shared>(vec)) {} + + output_adapter(std::basic_ostream& s) + : oa(std::make_shared>(s)) {} + + output_adapter(StringType& s) + : oa(std::make_shared>(s)) {} + + operator output_adapter_t() + { + return oa; + } + + private: + output_adapter_t oa = nullptr; +}; +} // namespace detail +} // namespace nlohmann + + namespace nlohmann { namespace detail @@ -9826,6 +10755,8 @@ class binary_writer } // namespace detail } // namespace nlohmann +// #include + // #include @@ -9843,8 +10774,6 @@ class binary_writer #include // is_same #include // move -// #include - // #include @@ -10944,6 +11873,8 @@ char* to_chars(char* first, const char* last, FloatType value) } // namespace detail } // namespace nlohmann +// #include + // #include // #include @@ -11211,6 +12142,11 @@ class serializer o->write_characters("null", 4); return; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } } @@ -11407,6 +12343,11 @@ class serializer state = UTF8_ACCEPT; break; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } break; } @@ -11467,6 +12408,11 @@ class serializer } break; } + + default: + { + assert(false); // LCOV_EXCL_LINE + } } } } @@ -11763,908 +12709,9 @@ class serializer } // namespace detail } // namespace nlohmann -// #include - - -#include -#include - -// #include - - -namespace nlohmann -{ -namespace detail -{ -template -class json_ref -{ - public: - using value_type = BasicJsonType; - - json_ref(value_type&& value) - : owned_value(std::move(value)), value_ref(&owned_value), is_rvalue(true) - {} - - json_ref(const value_type& value) - : value_ref(const_cast(&value)), is_rvalue(false) - {} - - json_ref(std::initializer_list init) - : owned_value(init), value_ref(&owned_value), is_rvalue(true) - {} - - template < - class... Args, - enable_if_t::value, int> = 0 > - json_ref(Args && ... args) - : owned_value(std::forward(args)...), value_ref(&owned_value), - is_rvalue(true) {} - - // class should be movable only - json_ref(json_ref&&) noexcept = default; - json_ref(const json_ref&) = delete; - json_ref& operator=(const json_ref&) = delete; - json_ref& operator=(json_ref&&) = delete; - ~json_ref() = default; - - value_type moved_or_copied() const - { - if (is_rvalue) - { - return std::move(*value_ref); - } - return *value_ref; - } - - value_type const& operator*() const - { - return *static_cast(value_ref); - } - - value_type const* operator->() const - { - return static_cast(value_ref); - } - - private: - mutable value_type owned_value = nullptr; - value_type* value_ref = nullptr; - const bool is_rvalue; -}; -} // namespace detail -} // namespace nlohmann - -// #include - - -#include // all_of -#include // assert -#include // accumulate -#include // string -#include // move -#include // vector - -// #include - -// #include - // #include - -namespace nlohmann -{ -template -class json_pointer -{ - // allow basic_json to access private members - NLOHMANN_BASIC_JSON_TPL_DECLARATION - friend class basic_json; - - public: - /*! - @brief create JSON pointer - - Create a JSON pointer according to the syntax described in - [Section 3 of RFC6901](https://tools.ietf.org/html/rfc6901#section-3). - - @param[in] s string representing the JSON pointer; if omitted, the empty - string is assumed which references the whole JSON value - - @throw parse_error.107 if the given JSON pointer @a s is nonempty and does - not begin with a slash (`/`); see example below - - @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s is - not followed by `0` (representing `~`) or `1` (representing `/`); see - example below - - @liveexample{The example shows the construction several valid JSON pointers - as well as the exceptional behavior.,json_pointer} - - @since version 2.0.0 - */ - explicit json_pointer(const std::string& s = "") - : reference_tokens(split(s)) - {} - - /*! - @brief return a string representation of the JSON pointer - - @invariant For each JSON pointer `ptr`, it holds: - @code {.cpp} - ptr == json_pointer(ptr.to_string()); - @endcode - - @return a string representation of the JSON pointer - - @liveexample{The example shows the result of `to_string`., - json_pointer__to_string} - - @since version 2.0.0 - */ - std::string to_string() const - { - return std::accumulate(reference_tokens.begin(), reference_tokens.end(), - std::string{}, - [](const std::string & a, const std::string & b) - { - return a + "/" + escape(b); - }); - } - - /// @copydoc to_string() - operator std::string() const - { - return to_string(); - } - - /*! - @brief append another JSON pointer at the end of this JSON pointer - */ - json_pointer& operator/=(const json_pointer& ptr) - { - reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); - return *this; - } - - /// @copydoc push_back(std::string&&) - json_pointer& operator/=(std::string token) - { - push_back(std::move(token)); - return *this; - } - - /// @copydoc operator/=(std::string) - json_pointer& operator/=(std::size_t array_index) - { - return *this /= std::to_string(array_index); - } - - /*! - @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer - */ - friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) - { - return json_pointer(left_ptr) /= right_ptr; - } - - /*! - @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer - */ - friend json_pointer operator/(const json_pointer& ptr, std::string token) - { - return json_pointer(ptr) /= std::move(token); - } - - /*! - @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer - */ - friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) - { - return json_pointer(lhs) /= array_index; - } - - /*! - @brief create a new JSON pointer that is the parent of this JSON pointer - */ - json_pointer parent_pointer() const - { - if (empty()) - { - return *this; - } - - json_pointer res = *this; - res.pop_back(); - return res; - } - - /*! - @param[in] s reference token to be converted into an array index - - @return integer representation of @a s - - @throw out_of_range.404 if string @a s could not be converted to an integer - */ - static int array_index(const std::string& s) - { - std::size_t processed_chars = 0; - const int res = std::stoi(s, &processed_chars); - - // check if the string was completely read - if (JSON_UNLIKELY(processed_chars != s.size())) - { - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); - } - - return res; - } - - /*! - @brief remove and return last reference token - @throw out_of_range.405 if JSON pointer has no parent - */ - std::string pop_back() - { - if (JSON_UNLIKELY(empty())) - { - JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); - } - - auto last = reference_tokens.back(); - reference_tokens.pop_back(); - return last; - } - - /*! - @brief append an unescaped token at the end of the reference pointer - */ - void push_back(const std::string& token) - { - reference_tokens.push_back(token); - } - - /// @copydoc push_back(const std::string&) - void push_back(std::string&& token) - { - reference_tokens.push_back(std::move(token)); - } - - /// return whether pointer points to the root document - bool empty() const noexcept - { - return reference_tokens.empty(); - } - - private: - json_pointer top() const - { - if (JSON_UNLIKELY(empty())) - { - JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); - } - - json_pointer result = *this; - result.reference_tokens = {reference_tokens[0]}; - return result; - } - - /*! - @brief create and return a reference to the pointed to value - - @complexity Linear in the number of reference tokens. - - @throw parse_error.109 if array index is not a number - @throw type_error.313 if value cannot be unflattened - */ - BasicJsonType& get_and_create(BasicJsonType& j) const - { - using size_type = typename BasicJsonType::size_type; - auto result = &j; - - // in case no reference tokens exist, return a reference to the JSON value - // j which will be overwritten by a primitive value - for (const auto& reference_token : reference_tokens) - { - switch (result->m_type) - { - case detail::value_t::null: - { - if (reference_token == "0") - { - // start a new array if reference token is 0 - result = &result->operator[](0); - } - else - { - // start a new object otherwise - result = &result->operator[](reference_token); - } - break; - } - - case detail::value_t::object: - { - // create an entry in the object - result = &result->operator[](reference_token); - break; - } - - case detail::value_t::array: - { - // create an entry in the array - JSON_TRY - { - result = &result->operator[](static_cast(array_index(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - /* - The following code is only reached if there exists a reference - token _and_ the current value is primitive. In this case, we have - an error situation, because primitive values may only occur as - single value; that is, with an empty list of reference tokens. - */ - default: - JSON_THROW(detail::type_error::create(313, "invalid value to unflatten")); - } - } - - return *result; - } - - /*! - @brief return a reference to the pointed to value - - @note This version does not throw if a value is not present, but tries to - create nested values instead. For instance, calling this function - with pointer `"/this/that"` on a null value is equivalent to calling - `operator[]("this").operator[]("that")` on that value, effectively - changing the null value to an object. - - @param[in] ptr a JSON value - - @return reference to the JSON value pointed to by the JSON pointer - - @complexity Linear in the length of the JSON pointer. - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - BasicJsonType& get_unchecked(BasicJsonType* ptr) const - { - using size_type = typename BasicJsonType::size_type; - for (const auto& reference_token : reference_tokens) - { - // convert null values to arrays or objects before continuing - if (ptr->m_type == detail::value_t::null) - { - // check if reference token is a number - const bool nums = - std::all_of(reference_token.begin(), reference_token.end(), - [](const char x) - { - return (x >= '0' and x <= '9'); - }); - - // change value to array for numbers or "-" or to object otherwise - *ptr = (nums or reference_token == "-") - ? detail::value_t::array - : detail::value_t::object; - } - - switch (ptr->m_type) - { - case detail::value_t::object: - { - // use unchecked object access - ptr = &ptr->operator[](reference_token); - break; - } - - case detail::value_t::array: - { - // error condition (cf. RFC 6901, Sect. 4) - if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) - { - JSON_THROW(detail::parse_error::create(106, 0, - "array index '" + reference_token + - "' must not begin with '0'")); - } - - if (reference_token == "-") - { - // explicitly treat "-" as index beyond the end - ptr = &ptr->operator[](ptr->m_value.array->size()); - } - else - { - // convert array index to number; unchecked access - JSON_TRY - { - ptr = &ptr->operator[]( - static_cast(array_index(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); - } - } - break; - } - - default: - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); - } - } - - return *ptr; - } - - /*! - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - BasicJsonType& get_checked(BasicJsonType* ptr) const - { - using size_type = typename BasicJsonType::size_type; - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case detail::value_t::object: - { - // note: at performs range check - ptr = &ptr->at(reference_token); - break; - } - - case detail::value_t::array: - { - if (JSON_UNLIKELY(reference_token == "-")) - { - // "-" always fails the range check - JSON_THROW(detail::out_of_range::create(402, - "array index '-' (" + std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) - { - JSON_THROW(detail::parse_error::create(106, 0, - "array index '" + reference_token + - "' must not begin with '0'")); - } - - // note: at performs range check - JSON_TRY - { - ptr = &ptr->at(static_cast(array_index(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); - } - } - - return *ptr; - } - - /*! - @brief return a const reference to the pointed to value - - @param[in] ptr a JSON value - - @return const reference to the JSON value pointed to by the JSON - pointer - - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - const BasicJsonType& get_unchecked(const BasicJsonType* ptr) const - { - using size_type = typename BasicJsonType::size_type; - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case detail::value_t::object: - { - // use unchecked object access - ptr = &ptr->operator[](reference_token); - break; - } - - case detail::value_t::array: - { - if (JSON_UNLIKELY(reference_token == "-")) - { - // "-" cannot be used for const access - JSON_THROW(detail::out_of_range::create(402, - "array index '-' (" + std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) - { - JSON_THROW(detail::parse_error::create(106, 0, - "array index '" + reference_token + - "' must not begin with '0'")); - } - - // use unchecked array access - JSON_TRY - { - ptr = &ptr->operator[]( - static_cast(array_index(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); - } - } - - return *ptr; - } - - /*! - @throw parse_error.106 if an array index begins with '0' - @throw parse_error.109 if an array index was not a number - @throw out_of_range.402 if the array index '-' is used - @throw out_of_range.404 if the JSON pointer can not be resolved - */ - const BasicJsonType& get_checked(const BasicJsonType* ptr) const - { - using size_type = typename BasicJsonType::size_type; - for (const auto& reference_token : reference_tokens) - { - switch (ptr->m_type) - { - case detail::value_t::object: - { - // note: at performs range check - ptr = &ptr->at(reference_token); - break; - } - - case detail::value_t::array: - { - if (JSON_UNLIKELY(reference_token == "-")) - { - // "-" always fails the range check - JSON_THROW(detail::out_of_range::create(402, - "array index '-' (" + std::to_string(ptr->m_value.array->size()) + - ") is out of range")); - } - - // error condition (cf. RFC 6901, Sect. 4) - if (JSON_UNLIKELY(reference_token.size() > 1 and reference_token[0] == '0')) - { - JSON_THROW(detail::parse_error::create(106, 0, - "array index '" + reference_token + - "' must not begin with '0'")); - } - - // note: at performs range check - JSON_TRY - { - ptr = &ptr->at(static_cast(array_index(reference_token))); - } - JSON_CATCH(std::invalid_argument&) - { - JSON_THROW(detail::parse_error::create(109, 0, "array index '" + reference_token + "' is not a number")); - } - break; - } - - default: - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'")); - } - } - - return *ptr; - } - - /*! - @brief split the string input to reference tokens - - @note This function is only called by the json_pointer constructor. - All exceptions below are documented there. - - @throw parse_error.107 if the pointer is not empty or begins with '/' - @throw parse_error.108 if character '~' is not followed by '0' or '1' - */ - static std::vector split(const std::string& reference_string) - { - std::vector result; - - // special case: empty reference string -> no reference tokens - if (reference_string.empty()) - { - return result; - } - - // check if nonempty reference string begins with slash - if (JSON_UNLIKELY(reference_string[0] != '/')) - { - JSON_THROW(detail::parse_error::create(107, 1, - "JSON pointer must be empty or begin with '/' - was: '" + - reference_string + "'")); - } - - // extract the reference tokens: - // - slash: position of the last read slash (or end of string) - // - start: position after the previous slash - for ( - // search for the first slash after the first character - std::size_t slash = reference_string.find_first_of('/', 1), - // set the beginning of the first reference token - start = 1; - // we can stop if start == 0 (if slash == std::string::npos) - start != 0; - // set the beginning of the next reference token - // (will eventually be 0 if slash == std::string::npos) - start = (slash == std::string::npos) ? 0 : slash + 1, - // find next slash - slash = reference_string.find_first_of('/', start)) - { - // use the text between the beginning of the reference token - // (start) and the last slash (slash). - auto reference_token = reference_string.substr(start, slash - start); - - // check reference tokens are properly escaped - for (std::size_t pos = reference_token.find_first_of('~'); - pos != std::string::npos; - pos = reference_token.find_first_of('~', pos + 1)) - { - assert(reference_token[pos] == '~'); - - // ~ must be followed by 0 or 1 - if (JSON_UNLIKELY(pos == reference_token.size() - 1 or - (reference_token[pos + 1] != '0' and - reference_token[pos + 1] != '1'))) - { - JSON_THROW(detail::parse_error::create(108, 0, "escape character '~' must be followed with '0' or '1'")); - } - } - - // finally, store the reference token - unescape(reference_token); - result.push_back(reference_token); - } - - return result; - } - - /*! - @brief replace all occurrences of a substring by another string - - @param[in,out] s the string to manipulate; changed so that all - occurrences of @a f are replaced with @a t - @param[in] f the substring to replace with @a t - @param[in] t the string to replace @a f - - @pre The search string @a f must not be empty. **This precondition is - enforced with an assertion.** - - @since version 2.0.0 - */ - static void replace_substring(std::string& s, const std::string& f, - const std::string& t) - { - assert(not f.empty()); - for (auto pos = s.find(f); // find first occurrence of f - pos != std::string::npos; // make sure f was found - s.replace(pos, f.size(), t), // replace with t, and - pos = s.find(f, pos + t.size())) // find next occurrence of f - {} - } - - /// escape "~" to "~0" and "/" to "~1" - static std::string escape(std::string s) - { - replace_substring(s, "~", "~0"); - replace_substring(s, "/", "~1"); - return s; - } - - /// unescape "~1" to tilde and "~0" to slash (order is important!) - static void unescape(std::string& s) - { - replace_substring(s, "~1", "/"); - replace_substring(s, "~0", "~"); - } - - /*! - @param[in] reference_string the reference string to the current value - @param[in] value the value to consider - @param[in,out] result the result object to insert values to - - @note Empty objects or arrays are flattened to `null`. - */ - static void flatten(const std::string& reference_string, - const BasicJsonType& value, - BasicJsonType& result) - { - switch (value.m_type) - { - case detail::value_t::array: - { - if (value.m_value.array->empty()) - { - // flatten empty array as null - result[reference_string] = nullptr; - } - else - { - // iterate array and use index as reference string - for (std::size_t i = 0; i < value.m_value.array->size(); ++i) - { - flatten(reference_string + "/" + std::to_string(i), - value.m_value.array->operator[](i), result); - } - } - break; - } - - case detail::value_t::object: - { - if (value.m_value.object->empty()) - { - // flatten empty object as null - result[reference_string] = nullptr; - } - else - { - // iterate object and use keys as reference string - for (const auto& element : *value.m_value.object) - { - flatten(reference_string + "/" + escape(element.first), element.second, result); - } - } - break; - } - - default: - { - // add primitive value with its reference string - result[reference_string] = value; - break; - } - } - } - - /*! - @param[in] value flattened JSON - - @return unflattened JSON - - @throw parse_error.109 if array index is not a number - @throw type_error.314 if value is not an object - @throw type_error.315 if object values are not primitive - @throw type_error.313 if value cannot be unflattened - */ - static BasicJsonType - unflatten(const BasicJsonType& value) - { - if (JSON_UNLIKELY(not value.is_object())) - { - JSON_THROW(detail::type_error::create(314, "only objects can be unflattened")); - } - - BasicJsonType result; - - // iterate the JSON object values - for (const auto& element : *value.m_value.object) - { - if (JSON_UNLIKELY(not element.second.is_primitive())) - { - JSON_THROW(detail::type_error::create(315, "values in object must be primitive")); - } - - // assign value to reference pointed to by JSON pointer; Note that if - // the JSON pointer is "" (i.e., points to the whole value), function - // get_and_create returns a reference to result itself. An assignment - // will then create a primitive value. - json_pointer(element.first).get_and_create(result) = element.second; - } - - return result; - } - - friend bool operator==(json_pointer const& lhs, - json_pointer const& rhs) noexcept - { - return (lhs.reference_tokens == rhs.reference_tokens); - } - - friend bool operator!=(json_pointer const& lhs, - json_pointer const& rhs) noexcept - { - return not (lhs == rhs); - } - - /// the reference tokens - std::vector reference_tokens; -}; -} // namespace nlohmann - -// #include - - -#include - -// #include - -// #include - - -namespace nlohmann -{ - -template -struct adl_serializer -{ - /*! - @brief convert a JSON value to any value type - - This function is usually called by the `get()` function of the - @ref basic_json class (either explicit or via conversion operators). - - @param[in] j JSON value to read from - @param[in,out] val value to write to - */ - template - static auto from_json(BasicJsonType&& j, ValueType& val) noexcept( - noexcept(::nlohmann::from_json(std::forward(j), val))) - -> decltype(::nlohmann::from_json(std::forward(j), val), void()) - { - ::nlohmann::from_json(std::forward(j), val); - } - - /*! - @brief convert any value type to a JSON value - - This function is usually called by the constructors of the @ref basic_json - class. - - @param[in,out] j JSON value to write to - @param[in] val value to read from - */ - template - static auto to_json(BasicJsonType& j, ValueType&& val) noexcept( - noexcept(::nlohmann::to_json(j, std::forward(val)))) - -> decltype(::nlohmann::to_json(j, std::forward(val)), void()) - { - ::nlohmann::to_json(j, std::forward(val)); - } -}; - -} // namespace nlohmann +// #include /*! @@ -13920,6 +13967,8 @@ class basic_json case value_t::discarded: m_type = value_t::discarded; break; + default: + assert(false); // LCOV_EXCL_LINE } assert_invariant(); } @@ -20290,7 +20339,7 @@ class basic_json break; } - case patch_operations::invalid: + default: { // op must be "add", "remove", "replace", "move", "copy", or // "test" diff --git a/test/src/unit-alt-string.cpp b/test/src/unit-alt-string.cpp index 9db580b4..9e0d4f94 100644 --- a/test/src/unit-alt-string.cpp +++ b/test/src/unit-alt-string.cpp @@ -149,7 +149,7 @@ class alt_string } private: - std::string str_impl; + std::string str_impl {}; friend bool ::operator<(const char*, const alt_string&); }; diff --git a/test/src/unit-class_parser.cpp b/test/src/unit-class_parser.cpp index 2d2929ee..42f6f497 100644 --- a/test/src/unit-class_parser.cpp +++ b/test/src/unit-class_parser.cpp @@ -125,7 +125,7 @@ class SaxEventLogger return false; } - std::vector events; + std::vector events {}; bool errored = false; }; diff --git a/test/src/unit-deserialization.cpp b/test/src/unit-deserialization.cpp index f92241ff..cd74b5e7 100644 --- a/test/src/unit-deserialization.cpp +++ b/test/src/unit-deserialization.cpp @@ -123,7 +123,7 @@ struct SaxEventLogger : public nlohmann::json_sax return false; } - std::vector events; + std::vector events {}; }; struct SaxEventLoggerExitAfterStartObject : public SaxEventLogger diff --git a/test/src/unit-regression.cpp b/test/src/unit-regression.cpp index 822cac96..930107bf 100644 --- a/test/src/unit-regression.cpp +++ b/test/src/unit-regression.cpp @@ -124,8 +124,10 @@ struct nocopy struct Data { - std::string a; - std::string b; + Data() = default; + Data(const std::string& a_, const std::string b_) : a(a_), b(b_) {} + std::string a {}; + std::string b {}; }; void from_json(const json& j, Data& data) @@ -1701,7 +1703,7 @@ TEST_CASE("regression tests") std::map expected { - {"1", {"testa_1", "testb_1" }}, + {"1", {"testa_1", "testb_1"}}, {"2", {"testa_2", "testb_2"}}, {"3", {"testa_3", "testb_3"}}, }; From 9f26dac9b3465e5933017ae092e6092e873f50ba Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 12:02:05 +0100 Subject: [PATCH 038/220] :arrow_up: updated Doxyfile --- doc/Doxyfile | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/doc/Doxyfile b/doc/Doxyfile index 10390b76..148ff12b 100644 --- a/doc/Doxyfile +++ b/doc/Doxyfile @@ -1,4 +1,4 @@ -# Doxyfile 1.8.15 +# Doxyfile 1.8.16 #--------------------------------------------------------------------------- # Project related configuration options @@ -38,6 +38,7 @@ OPTIMIZE_OUTPUT_FOR_C = NO OPTIMIZE_OUTPUT_JAVA = NO OPTIMIZE_FOR_FORTRAN = NO OPTIMIZE_OUTPUT_VHDL = NO +OPTIMIZE_OUTPUT_SLICE = NO EXTENSION_MAPPING = MARKDOWN_SUPPORT = YES TOC_INCLUDE_HEADINGS = 0 @@ -141,7 +142,7 @@ USE_HTAGS = NO VERBATIM_HEADERS = NO CLANG_ASSISTED_PARSING = YES CLANG_OPTIONS = -std=c++11 -CLANG_COMPILATION_DATABASE_PATH = 0 +CLANG_DATABASE_PATH = #--------------------------------------------------------------------------- # Configuration options related to the alphabetical class index #--------------------------------------------------------------------------- @@ -214,6 +215,7 @@ GENERATE_LATEX = NO LATEX_OUTPUT = latex LATEX_CMD_NAME = latex MAKEINDEX_CMD_NAME = makeindex +LATEX_MAKEINDEX_CMD = \makeindex COMPACT_LATEX = NO PAPER_TYPE = a4 EXTRA_PACKAGES = @@ -228,6 +230,7 @@ LATEX_HIDE_INDICES = NO LATEX_SOURCE_CODE = NO LATEX_BIB_STYLE = plain LATEX_TIMESTAMP = NO +LATEX_EMOJI_DIRECTORY = #--------------------------------------------------------------------------- # Configuration options related to the RTF output #--------------------------------------------------------------------------- @@ -252,6 +255,7 @@ MAN_LINKS = NO GENERATE_XML = YES XML_OUTPUT = xml XML_PROGRAMLISTING = YES +XML_NS_MEMB_FILE_SCOPE = NO #--------------------------------------------------------------------------- # Configuration options related to the DOCBOOK output #--------------------------------------------------------------------------- From df0f7f2b5d2b172a856e61e49888f7aefde0572c Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 12:02:23 +0100 Subject: [PATCH 039/220] :construction_worker: overworked clang-tidy target --- .clang-tidy | 23 +++++++++++++++++++++++ Makefile | 9 ++++----- 2 files changed, 27 insertions(+), 5 deletions(-) create mode 100644 .clang-tidy diff --git a/.clang-tidy b/.clang-tidy new file mode 100644 index 00000000..0a04d286 --- /dev/null +++ b/.clang-tidy @@ -0,0 +1,23 @@ +Checks: '-*, + bugprone-*, + cert-*, + clang-analyzer-*, + google-*, + -google-runtime-references, + hicpp-*, + -hicpp-no-array-decay, + -hicpp-uppercase-literal-suffix, + misc-*, + -misc-non-private-member-variables-in-classes, + llvm-*, + -llvm-header-guard, + modernize-*, + performance-*, + portability-*, + readability-*, + -readability-magic-numbers, + -readability-uppercase-literal-suffix' + +CheckOptions: + - key: hicpp-special-member-functions.AllowSoleDefaultDtor + value: 1 diff --git a/Makefile b/Makefile index 7d2bfb91..7d7ce5f6 100644 --- a/Makefile +++ b/Makefile @@ -192,9 +192,11 @@ pedantic_gcc: -Wdouble-promotion \ -Wduplicated-branches \ -Wduplicated-cond \ + -Weffc++ \ -Wempty-body \ -Wendif-labels \ -Wenum-compare \ + -Wexpansion-to-defined \ -Werror \ -Wextra \ -Wextra-semi \ @@ -249,13 +251,9 @@ pedantic_gcc: -Wmultistatement-macros \ -Wnarrowing \ -Wno-deprecated-declarations \ - -Wno-effc++ \ - -Wno-expansion-to-defined \ - -Wno-ignored-qualifiers \ -Wno-long-long \ -Wno-namespaces \ -Wno-padded \ - -Wno-switch-default \ -Wno-switch-enum \ -Wno-system-headers \ -Wno-templates \ @@ -332,6 +330,7 @@ pedantic_gcc: -Wsuggest-override \ -Wswitch \ -Wswitch-bool \ + -Wswitch-default \ -Wswitch-unreachable \ -Wsync-nand \ -Wsynth \ @@ -457,7 +456,7 @@ cpplint: third_party/cpplint/cpplint.py --filter=-whitespace,-legal,-readability/alt_tokens,-runtime/references,-runtime/explicit --quiet --recursive include clang_tidy: - $(COMPILER_DIR)/clang-tidy -checks='-*, readability-*, -readability-magic-numbers, -readability-uppercase-literal-suffix, modernize-*, bugprone-*, performance-*, clang-analyzer-*, portability-*, cert-*, hicpp-*, -hicpp-no-array-decay, -hicpp-uppercase-literal-suffix, google-*, -google-runtime-references' $(SRCS) -- -Iinclude -std=c++11 + $(COMPILER_DIR)/clang-tidy $(SRCS) -- -Iinclude -std=c++11 pvs_studio: rm -fr pvs_studio_build From 56f6d1d68e618649aa56b58d1602a139f8000d92 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 15:20:22 +0100 Subject: [PATCH 040/220] :green_heart: fix CI and #1521 --- include/nlohmann/detail/input/binary_reader.hpp | 4 ++-- include/nlohmann/detail/input/json_sax.hpp | 4 ++-- include/nlohmann/detail/json_ref.hpp | 2 +- include/nlohmann/detail/output/serializer.hpp | 2 +- single_include/nlohmann/json.hpp | 12 ++++++------ 5 files changed, 12 insertions(+), 12 deletions(-) diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index cded3b54..3ac10990 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -54,9 +54,9 @@ class binary_reader // make class move-only binary_reader(const binary_reader&) = delete; - binary_reader(binary_reader&&) noexcept = default; + binary_reader(binary_reader&&) = default; binary_reader& operator=(const binary_reader&) = delete; - binary_reader& operator=(binary_reader&&) noexcept = default; + binary_reader& operator=(binary_reader&&) = default; ~binary_reader() = default; /*! diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index 097b33e5..f82255af 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -161,9 +161,9 @@ class json_sax_dom_parser // make class move-only json_sax_dom_parser(const json_sax_dom_parser&) = delete; - json_sax_dom_parser(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser(json_sax_dom_parser&&) = default; json_sax_dom_parser& operator=(const json_sax_dom_parser&) = delete; - json_sax_dom_parser& operator=(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser& operator=(json_sax_dom_parser&&) = default; ~json_sax_dom_parser() = default; bool null() diff --git a/include/nlohmann/detail/json_ref.hpp b/include/nlohmann/detail/json_ref.hpp index 26da3964..c8dec733 100644 --- a/include/nlohmann/detail/json_ref.hpp +++ b/include/nlohmann/detail/json_ref.hpp @@ -35,7 +35,7 @@ class json_ref is_rvalue(true) {} // class should be movable only - json_ref(json_ref&&) noexcept = default; + json_ref(json_ref&&) = default; json_ref(const json_ref&) = delete; json_ref& operator=(const json_ref&) = delete; json_ref& operator=(json_ref&&) = delete; diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index 430eff62..31174eac 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -626,7 +626,7 @@ class serializer } // use a pointer to fill the buffer - auto buffer_ptr = begin(number_buffer); + auto buffer_ptr = number_buffer.begin(); const bool is_negative = std::is_same::value and not(x >= 0); // see issue #755 number_unsigned_t abs_value; diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 8e677438..338d2b0d 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -2839,9 +2839,9 @@ class json_sax_dom_parser // make class move-only json_sax_dom_parser(const json_sax_dom_parser&) = delete; - json_sax_dom_parser(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser(json_sax_dom_parser&&) = default; json_sax_dom_parser& operator=(const json_sax_dom_parser&) = delete; - json_sax_dom_parser& operator=(json_sax_dom_parser&&) noexcept = default; + json_sax_dom_parser& operator=(json_sax_dom_parser&&) = default; ~json_sax_dom_parser() = default; bool null() @@ -3579,9 +3579,9 @@ class binary_reader // make class move-only binary_reader(const binary_reader&) = delete; - binary_reader(binary_reader&&) noexcept = default; + binary_reader(binary_reader&&) = default; binary_reader& operator=(const binary_reader&) = delete; - binary_reader& operator=(binary_reader&&) noexcept = default; + binary_reader& operator=(binary_reader&&) = default; ~binary_reader() = default; /*! @@ -9258,7 +9258,7 @@ class json_ref is_rvalue(true) {} // class should be movable only - json_ref(json_ref&&) noexcept = default; + json_ref(json_ref&&) = default; json_ref(const json_ref&) = delete; json_ref& operator=(const json_ref&) = delete; json_ref& operator=(json_ref&&) = delete; @@ -12490,7 +12490,7 @@ class serializer } // use a pointer to fill the buffer - auto buffer_ptr = begin(number_buffer); + auto buffer_ptr = number_buffer.begin(); const bool is_negative = std::is_same::value and not(x >= 0); // see issue #755 number_unsigned_t abs_value; From 22c733e6fe7b166f5c69a949189e3e9a555721c4 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Sun, 17 Mar 2019 22:25:18 +0100 Subject: [PATCH 041/220] :memo: added documentation --- doc/examples/contains.cpp | 17 ++++ doc/examples/contains.link | 1 + doc/examples/contains.output | 3 + doc/examples/json_pointer__empty.cpp | 20 ++++ doc/examples/json_pointer__empty.link | 1 + doc/examples/json_pointer__empty.output | 4 + doc/examples/json_pointer__parent_pointer.cpp | 18 ++++ .../json_pointer__parent_pointer.link | 1 + .../json_pointer__parent_pointer.output | 3 + doc/examples/json_pointer__push_back.cpp | 21 ++++ doc/examples/json_pointer__push_back.link | 1 + doc/examples/json_pointer__push_back.output | 4 + doc/index.md | 1 + include/nlohmann/detail/exceptions.hpp | 2 +- .../nlohmann/detail/input/input_adapters.hpp | 4 +- include/nlohmann/detail/json_pointer.hpp | 85 +++++++++++----- include/nlohmann/json.hpp | 6 ++ single_include/nlohmann/json.hpp | 97 +++++++++++++------ 18 files changed, 233 insertions(+), 56 deletions(-) create mode 100644 doc/examples/contains.cpp create mode 100644 doc/examples/contains.link create mode 100644 doc/examples/contains.output create mode 100644 doc/examples/json_pointer__empty.cpp create mode 100644 doc/examples/json_pointer__empty.link create mode 100644 doc/examples/json_pointer__empty.output create mode 100644 doc/examples/json_pointer__parent_pointer.cpp create mode 100644 doc/examples/json_pointer__parent_pointer.link create mode 100644 doc/examples/json_pointer__parent_pointer.output create mode 100644 doc/examples/json_pointer__push_back.cpp create mode 100644 doc/examples/json_pointer__push_back.link create mode 100644 doc/examples/json_pointer__push_back.output diff --git a/doc/examples/contains.cpp b/doc/examples/contains.cpp new file mode 100644 index 00000000..df8201c3 --- /dev/null +++ b/doc/examples/contains.cpp @@ -0,0 +1,17 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // create some JSON values + json j_object = R"( {"key": "value"} )"_json; + json j_array = R"( [1, 2, 3] )"_json; + + // call contains + std::cout << std::boolalpha << + "j_object contains 'key': " << j_object.contains("key") << '\n' << + "j_object contains 'another': " << j_object.contains("another") << '\n' << + "j_array contains 'key': " << j_array.contains("key") << std::endl; +} diff --git a/doc/examples/contains.link b/doc/examples/contains.link new file mode 100644 index 00000000..c66676a6 --- /dev/null +++ b/doc/examples/contains.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/contains.output b/doc/examples/contains.output new file mode 100644 index 00000000..14ad177b --- /dev/null +++ b/doc/examples/contains.output @@ -0,0 +1,3 @@ +j_object contains 'key': true +j_object contains 'another': false +j_array contains 'key': false diff --git a/doc/examples/json_pointer__empty.cpp b/doc/examples/json_pointer__empty.cpp new file mode 100644 index 00000000..5daaadc6 --- /dev/null +++ b/doc/examples/json_pointer__empty.cpp @@ -0,0 +1,20 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // different JSON Pointers + json::json_pointer ptr0; + json::json_pointer ptr1(""); + json::json_pointer ptr2("/foo"); + json::json_pointer ptr3("/foo/0"); + + // call empty() + std::cout << std::boolalpha + << ptr0 << ": " << ptr0.empty() << '\n' + << ptr1 << ": " << ptr1.empty() << '\n' + << ptr2 << ": " << ptr2.empty() << '\n' + << ptr3 << ": " << ptr3.empty() << std::endl; +} diff --git a/doc/examples/json_pointer__empty.link b/doc/examples/json_pointer__empty.link new file mode 100644 index 00000000..e3759120 --- /dev/null +++ b/doc/examples/json_pointer__empty.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__empty.output b/doc/examples/json_pointer__empty.output new file mode 100644 index 00000000..a7ee49c1 --- /dev/null +++ b/doc/examples/json_pointer__empty.output @@ -0,0 +1,4 @@ +"": true +"": true +"/foo": false +"/foo/0": false diff --git a/doc/examples/json_pointer__parent_pointer.cpp b/doc/examples/json_pointer__parent_pointer.cpp new file mode 100644 index 00000000..6021463a --- /dev/null +++ b/doc/examples/json_pointer__parent_pointer.cpp @@ -0,0 +1,18 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // different JSON Pointers + json::json_pointer ptr1(""); + json::json_pointer ptr2("/foo"); + json::json_pointer ptr3("/foo/0"); + + // call parent_pointer() + std::cout << std::boolalpha + << "parent of " << ptr1 << " is " << ptr1.parent_pointer() << '\n' + << "parent of " << ptr2 << " is " << ptr2.parent_pointer() << '\n' + << "parent of " << ptr3 << " is " << ptr3.parent_pointer() << std::endl; +} diff --git a/doc/examples/json_pointer__parent_pointer.link b/doc/examples/json_pointer__parent_pointer.link new file mode 100644 index 00000000..4aa6327e --- /dev/null +++ b/doc/examples/json_pointer__parent_pointer.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__parent_pointer.output b/doc/examples/json_pointer__parent_pointer.output new file mode 100644 index 00000000..4cc6f3f1 --- /dev/null +++ b/doc/examples/json_pointer__parent_pointer.output @@ -0,0 +1,3 @@ +parent of "" is "" +parent of "/foo" is "" +parent of "/foo/0" is "/foo" diff --git a/doc/examples/json_pointer__push_back.cpp b/doc/examples/json_pointer__push_back.cpp new file mode 100644 index 00000000..d6536b3f --- /dev/null +++ b/doc/examples/json_pointer__push_back.cpp @@ -0,0 +1,21 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // create empty JSON Pointer + json::json_pointer ptr; + std::cout << ptr << '\n'; + + // call push_back() + ptr.push_back("foo"); + std::cout << ptr << '\n'; + + ptr.push_back("0"); + std::cout << ptr << '\n'; + + ptr.push_back("bar"); + std::cout << ptr << '\n'; +} diff --git a/doc/examples/json_pointer__push_back.link b/doc/examples/json_pointer__push_back.link new file mode 100644 index 00000000..d0cadd7c --- /dev/null +++ b/doc/examples/json_pointer__push_back.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__push_back.output b/doc/examples/json_pointer__push_back.output new file mode 100644 index 00000000..92c019cd --- /dev/null +++ b/doc/examples/json_pointer__push_back.output @@ -0,0 +1,4 @@ +"" +"/foo" +"/foo/0" +"/foo/0/bar" diff --git a/doc/index.md b/doc/index.md index 1266a273..ea6dc6ae 100644 --- a/doc/index.md +++ b/doc/index.md @@ -57,6 +57,7 @@ These pages contain the API documentation of JSON for Modern C++, a C++11 header - @link nlohmann::basic_json::number_integer_t signed integers @endlink - @link nlohmann::basic_json::number_unsigned_t unsigned integers @endlink - @link nlohmann::basic_json::number_float_t floating-point @endlink +- @link nlohmann::json_pointer JSON Pointer @endlink # Container function overview diff --git a/include/nlohmann/detail/exceptions.hpp b/include/nlohmann/detail/exceptions.hpp index 5a73dedb..519a5975 100644 --- a/include/nlohmann/detail/exceptions.hpp +++ b/include/nlohmann/detail/exceptions.hpp @@ -223,7 +223,7 @@ name / id | example message | description ----------------------------- | --------------- | ------------------------- json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t &. json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. diff --git a/include/nlohmann/detail/input/input_adapters.hpp b/include/nlohmann/detail/input/input_adapters.hpp index 0374731f..8730748c 100644 --- a/include/nlohmann/detail/input/input_adapters.hpp +++ b/include/nlohmann/detail/input/input_adapters.hpp @@ -61,9 +61,9 @@ class file_input_adapter : public input_adapter_protocol // make class move-only file_input_adapter(const file_input_adapter&) = delete; - file_input_adapter(file_input_adapter&&) noexcept = default; + file_input_adapter(file_input_adapter&&) = default; file_input_adapter& operator=(const file_input_adapter&) = delete; - file_input_adapter& operator=(file_input_adapter&&) noexcept = default; + file_input_adapter& operator=(file_input_adapter&&) = default; ~file_input_adapter() override = default; std::char_traits::int_type get_character() noexcept override diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index a2db2357..3d44e267 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -82,7 +82,9 @@ class json_pointer */ json_pointer& operator/=(const json_pointer& ptr) { - reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); + reference_tokens.insert(reference_tokens.end(), + ptr.reference_tokens.begin(), + ptr.reference_tokens.end()); return *this; } @@ -102,7 +104,8 @@ class json_pointer /*! @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer */ - friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) + friend json_pointer operator/(const json_pointer& left_ptr, + const json_pointer& right_ptr) { return json_pointer(left_ptr) /= right_ptr; } @@ -124,7 +127,17 @@ class json_pointer } /*! - @brief create a new JSON pointer that is the parent of this JSON pointer + @brief returns the parent of this JSON pointer + + @return parent of this JSON pointer; in case this JSON pointer is the root, + the root itself is returned + + @complexity Constant. + + @liveexample{The example shows the result of `parent_pointer` for different + JSON Pointers.,json_pointer__parent_pointer} + + @since version 3.6.0 */ json_pointer parent_pointer() const { @@ -138,27 +151,6 @@ class json_pointer return res; } - /*! - @param[in] s reference token to be converted into an array index - - @return integer representation of @a s - - @throw out_of_range.404 if string @a s could not be converted to an integer - */ - static int array_index(const std::string& s) - { - std::size_t processed_chars = 0; - const int res = std::stoi(s, &processed_chars); - - // check if the string was completely read - if (JSON_UNLIKELY(processed_chars != s.size())) - { - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); - } - - return res; - } - /*! @brief remove and return last reference token @throw out_of_range.405 if JSON pointer has no parent @@ -177,6 +169,15 @@ class json_pointer /*! @brief append an unescaped token at the end of the reference pointer + + @param[in] token token to add + + @complexity Amortized constant. + + @liveexample{The example shows the result of `push_back` for different + JSON Pointers.,json_pointer__push_back} + + @since version 0.6.0 */ void push_back(const std::string& token) { @@ -189,13 +190,47 @@ class json_pointer reference_tokens.push_back(std::move(token)); } - /// return whether pointer points to the root document + /*! + @brief return whether pointer points to the root document + + @return true iff the JSON pointer points to the root document + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example shows the result of `empty` for different JSON + Pointers.,json_pointer__empty} + + @since version 3.6.0 + */ bool empty() const noexcept { return reference_tokens.empty(); } private: + /*! + @param[in] s reference token to be converted into an array index + + @return integer representation of @a s + + @throw out_of_range.404 if string @a s could not be converted to an integer + */ + static int array_index(const std::string& s) + { + std::size_t processed_chars = 0; + const int res = std::stoi(s, &processed_chars); + + // check if the string was completely read + if (JSON_UNLIKELY(processed_chars != s.size())) + { + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); + } + + return res; + } + json_pointer top() const { if (JSON_UNLIKELY(empty())) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 769126e5..6bc13727 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -3901,6 +3901,8 @@ class basic_json @liveexample{The example shows how `find()` is used.,find__key_type} + @sa @ref contains(KeyT&&) const -- checks whether a key exists + @since version 1.0.0 */ template @@ -3979,6 +3981,10 @@ class basic_json @complexity Logarithmic in the size of the JSON object. + @liveexample{The following code shows an example for `contains()`.,contains} + + @sa @ref find(KeyT&&) -- returns an iterator to an object element + @since version 3.6.0 */ template diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 338d2b0d..f5b3aa7c 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -323,7 +323,7 @@ name / id | example message | description ----------------------------- | --------------- | ------------------------- json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. -json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t&. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t &. json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. @@ -2296,9 +2296,9 @@ class file_input_adapter : public input_adapter_protocol // make class move-only file_input_adapter(const file_input_adapter&) = delete; - file_input_adapter(file_input_adapter&&) noexcept = default; + file_input_adapter(file_input_adapter&&) = default; file_input_adapter& operator=(const file_input_adapter&) = delete; - file_input_adapter& operator=(file_input_adapter&&) noexcept = default; + file_input_adapter& operator=(file_input_adapter&&) = default; ~file_input_adapter() override = default; std::char_traits::int_type get_character() noexcept override @@ -8529,7 +8529,9 @@ class json_pointer */ json_pointer& operator/=(const json_pointer& ptr) { - reference_tokens.insert(reference_tokens.end(), ptr.reference_tokens.begin(), ptr.reference_tokens.end()); + reference_tokens.insert(reference_tokens.end(), + ptr.reference_tokens.begin(), + ptr.reference_tokens.end()); return *this; } @@ -8549,7 +8551,8 @@ class json_pointer /*! @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer */ - friend json_pointer operator/(const json_pointer& left_ptr, const json_pointer& right_ptr) + friend json_pointer operator/(const json_pointer& left_ptr, + const json_pointer& right_ptr) { return json_pointer(left_ptr) /= right_ptr; } @@ -8571,7 +8574,17 @@ class json_pointer } /*! - @brief create a new JSON pointer that is the parent of this JSON pointer + @brief returns the parent of this JSON pointer + + @return parent of this JSON pointer; in case this JSON pointer is the root, + the root itself is returned + + @complexity Constant. + + @liveexample{The example shows the result of `parent_pointer` for different + JSON Pointers.,json_pointer__parent_pointer} + + @since version 3.6.0 */ json_pointer parent_pointer() const { @@ -8585,27 +8598,6 @@ class json_pointer return res; } - /*! - @param[in] s reference token to be converted into an array index - - @return integer representation of @a s - - @throw out_of_range.404 if string @a s could not be converted to an integer - */ - static int array_index(const std::string& s) - { - std::size_t processed_chars = 0; - const int res = std::stoi(s, &processed_chars); - - // check if the string was completely read - if (JSON_UNLIKELY(processed_chars != s.size())) - { - JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); - } - - return res; - } - /*! @brief remove and return last reference token @throw out_of_range.405 if JSON pointer has no parent @@ -8624,6 +8616,15 @@ class json_pointer /*! @brief append an unescaped token at the end of the reference pointer + + @param[in] token token to add + + @complexity Amortized constant. + + @liveexample{The example shows the result of `push_back` for different + JSON Pointers.,json_pointer__push_back} + + @since version 0.6.0 */ void push_back(const std::string& token) { @@ -8636,13 +8637,47 @@ class json_pointer reference_tokens.push_back(std::move(token)); } - /// return whether pointer points to the root document + /*! + @brief return whether pointer points to the root document + + @return true iff the JSON pointer points to the root document + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example shows the result of `empty` for different JSON + Pointers.,json_pointer__empty} + + @since version 3.6.0 + */ bool empty() const noexcept { return reference_tokens.empty(); } private: + /*! + @param[in] s reference token to be converted into an array index + + @return integer representation of @a s + + @throw out_of_range.404 if string @a s could not be converted to an integer + */ + static int array_index(const std::string& s) + { + std::size_t processed_chars = 0; + const int res = std::stoi(s, &processed_chars); + + // check if the string was completely read + if (JSON_UNLIKELY(processed_chars != s.size())) + { + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'")); + } + + return res; + } + json_pointer top() const { if (JSON_UNLIKELY(empty())) @@ -16543,6 +16578,8 @@ class basic_json @liveexample{The example shows how `find()` is used.,find__key_type} + @sa @ref contains(KeyT&&) const -- checks whether a key exists + @since version 1.0.0 */ template @@ -16621,6 +16658,10 @@ class basic_json @complexity Logarithmic in the size of the JSON object. + @liveexample{The following code shows an example for `contains()`.,contains} + + @sa @ref find(KeyT&&) -- returns an iterator to an object element + @since version 3.6.0 */ template From 9fc093c9e0a3e51676685cbd1b4019150953ccf9 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 18 Mar 2019 00:19:28 +0100 Subject: [PATCH 042/220] :construction_worker: added targets for infer and oclint --- Makefile | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/Makefile b/Makefile index 7d7ce5f6..086b07f1 100644 --- a/Makefile +++ b/Makefile @@ -466,6 +466,15 @@ pvs_studio: cd pvs_studio_build ; plog-converter -a'GA:1,2;64:1;CS' -t fullhtml PVS-Studio.log -o pvs open pvs_studio_build/pvs/index.html +infer: + rm -fr infer_build + mkdir infer_build + cd infer_build ; infer compile -- cmake .. ; infer run -- make -j 4 + +oclint: + oclint $(SRCS) -report-type html -enable-global-analysis -o oclint_report.html -max-priority-1=10000 -max-priority-2=10000 -max-priority-3=10000 -- -std=c++11 -Iinclude + open oclint_report.html + ########################################################################## # maintainer targets ########################################################################## From 8d3f4f21bcd846429bb38cdf3c91c3bd89e1f193 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 18 Mar 2019 13:53:48 +0100 Subject: [PATCH 043/220] :hammer: clean up --- .../nlohmann/detail/conversions/to_chars.hpp | 4 +- .../nlohmann/detail/input/binary_reader.hpp | 23 +- .../nlohmann/detail/input/input_adapters.hpp | 2 +- include/nlohmann/detail/input/json_sax.hpp | 85 ++-- include/nlohmann/detail/input/lexer.hpp | 10 +- include/nlohmann/detail/input/parser.hpp | 2 +- include/nlohmann/detail/json_pointer.hpp | 4 +- include/nlohmann/detail/macro_scope.hpp | 4 +- include/nlohmann/detail/output/serializer.hpp | 14 +- include/nlohmann/json.hpp | 324 ++++++------ single_include/nlohmann/json.hpp | 472 ++++++++---------- 11 files changed, 432 insertions(+), 512 deletions(-) diff --git a/include/nlohmann/detail/conversions/to_chars.hpp b/include/nlohmann/detail/conversions/to_chars.hpp index 198e422e..241dd372 100644 --- a/include/nlohmann/detail/conversions/to_chars.hpp +++ b/include/nlohmann/detail/conversions/to_chars.hpp @@ -204,7 +204,7 @@ boundaries compute_boundaries(FloatType value) const std::uint64_t E = bits >> (kPrecision - 1); const std::uint64_t F = bits & (kHiddenBit - 1); - const bool is_denormal = (E == 0); + const bool is_denormal = E == 0; const diyfp v = is_denormal ? diyfp(F, kMinExp) : diyfp(F + kHiddenBit, static_cast(E) - kBias); @@ -230,7 +230,7 @@ boundaries compute_boundaries(FloatType value) // -----------------+------+------+-------------+-------------+--- (B) // v- m- v m+ v+ - const bool lower_boundary_is_closer = (F == 0 and E > 1); + const bool lower_boundary_is_closer = F == 0 and E > 1; const diyfp m_plus = diyfp(2 * v.f + 1, v.e - 1); const diyfp m_minus = lower_boundary_is_closer ? diyfp(4 * v.f - 1, v.e - 2) // (B) diff --git a/include/nlohmann/detail/input/binary_reader.hpp b/include/nlohmann/detail/input/binary_reader.hpp index 3ac10990..edff2458 100644 --- a/include/nlohmann/detail/input/binary_reader.hpp +++ b/include/nlohmann/detail/input/binary_reader.hpp @@ -91,10 +91,8 @@ class binary_reader result = parse_ubjson_internal(); break; - // LCOV_EXCL_START - default: - assert(false); - // LCOV_EXCL_STOP + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } // strict mode: next byte must be EOF @@ -128,7 +126,7 @@ class binary_reader */ static constexpr bool little_endianess(int num = 1) noexcept { - return (*reinterpret_cast(&num) == 1); + return *reinterpret_cast(&num) == 1; } private: @@ -305,12 +303,9 @@ class binary_reader return false; } - if (not is_array) + if (not is_array and not sax->key(key)) { - if (not sax->key(key)) - { - return false; - } + return false; } if (JSON_UNLIKELY(not parse_bson_element_internal(element_type, element_type_parse_position))) @@ -1818,7 +1813,7 @@ class binary_reader int get() { ++chars_read; - return (current = ia->get_character()); + return current = ia->get_character(); } /*! @@ -1964,10 +1959,8 @@ class binary_reader error_msg += "BSON"; break; - // LCOV_EXCL_START - default: - assert(false); - // LCOV_EXCL_STOP + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } return error_msg + " " + context + ": " + detail; diff --git a/include/nlohmann/detail/input/input_adapters.hpp b/include/nlohmann/detail/input/input_adapters.hpp index 8730748c..5d9e200e 100644 --- a/include/nlohmann/detail/input/input_adapters.hpp +++ b/include/nlohmann/detail/input/input_adapters.hpp @@ -286,7 +286,7 @@ template class wide_string_input_adapter : public input_adapter_protocol { public: - explicit wide_string_input_adapter(const WideStringType& w) noexcept + explicit wide_string_input_adapter(const WideStringType& w) noexcept : str(w) {} diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index f82255af..fd1ba76c 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -303,12 +303,11 @@ class json_sax_dom_parser ref_stack.back()->m_value.array->emplace_back(std::forward(v)); return &(ref_stack.back()->m_value.array->back()); } - else - { - assert(object_element); - *object_element = BasicJsonType(std::forward(v)); - return object_element; - } + + assert(ref_stack.back()->is_object()) + assert(object_element); + *object_element = BasicJsonType(std::forward(v)); + return object_element; } /// the parsed JSON value @@ -395,13 +394,9 @@ class json_sax_dom_callback_parser ref_stack.push_back(val.second); // check object limit - if (ref_stack.back()) + if (ref_stack.back() and JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) { - if (JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) - { - JSON_THROW(out_of_range::create(408, - "excessive object size: " + std::to_string(len))); - } + JSON_THROW(out_of_range::create(408, "excessive object size: " + std::to_string(len))); } return true; @@ -426,13 +421,10 @@ class json_sax_dom_callback_parser bool end_object() { - if (ref_stack.back()) + if (ref_stack.back() and not callback(static_cast(ref_stack.size()) - 1, parse_event_t::object_end, *ref_stack.back())) { - if (not callback(static_cast(ref_stack.size()) - 1, parse_event_t::object_end, *ref_stack.back())) - { - // discard object - *ref_stack.back() = discarded; - } + // discard object + *ref_stack.back() = discarded; } assert(not ref_stack.empty()); @@ -440,18 +432,15 @@ class json_sax_dom_callback_parser ref_stack.pop_back(); keep_stack.pop_back(); - if (not ref_stack.empty() and ref_stack.back()) + if (not ref_stack.empty() and ref_stack.back() and ref_stack.back()->is_object()) { // remove discarded value - if (ref_stack.back()->is_object()) + for (auto it = ref_stack.back()->begin(); it != ref_stack.back()->end(); ++it) { - for (auto it = ref_stack.back()->begin(); it != ref_stack.back()->end(); ++it) + if (it->is_discarded()) { - if (it->is_discarded()) - { - ref_stack.back()->erase(it); - break; - } + ref_stack.back()->erase(it); + break; } } } @@ -468,13 +457,9 @@ class json_sax_dom_callback_parser ref_stack.push_back(val.second); // check array limit - if (ref_stack.back()) + if (ref_stack.back() and JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) { - if (JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) - { - JSON_THROW(out_of_range::create(408, - "excessive array size: " + std::to_string(len))); - } + JSON_THROW(out_of_range::create(408, "excessive array size: " + std::to_string(len))); } return true; @@ -500,12 +485,9 @@ class json_sax_dom_callback_parser keep_stack.pop_back(); // remove discarded value - if (not keep and not ref_stack.empty()) + if (not keep and not ref_stack.empty() and ref_stack.back()->is_array()) { - if (ref_stack.back()->is_array()) - { - ref_stack.back()->m_value.array->pop_back(); - } + ref_stack.back()->m_value.array->pop_back(); } return true; @@ -600,27 +582,28 @@ class json_sax_dom_callback_parser // we now only expect arrays and objects assert(ref_stack.back()->is_array() or ref_stack.back()->is_object()); + // array if (ref_stack.back()->is_array()) { ref_stack.back()->m_value.array->push_back(std::move(value)); return {true, &(ref_stack.back()->m_value.array->back())}; } - else + + // object + assert(ref_stack.back()->is_object()) + // check if we should store an element for the current key + assert(not key_keep_stack.empty()); + const bool store_element = key_keep_stack.back(); + key_keep_stack.pop_back(); + + if (not store_element) { - // check if we should store an element for the current key - assert(not key_keep_stack.empty()); - const bool store_element = key_keep_stack.back(); - key_keep_stack.pop_back(); - - if (not store_element) - { - return {false, nullptr}; - } - - assert(object_element); - *object_element = std::move(value); - return {true, object_element}; + return {false, nullptr}; } + + assert(object_element); + *object_element = std::move(value); + return {true, object_element}; } /// the parsed JSON value diff --git a/include/nlohmann/detail/input/lexer.hpp b/include/nlohmann/detail/input/lexer.hpp index d056b8f5..baee769e 100644 --- a/include/nlohmann/detail/input/lexer.hpp +++ b/include/nlohmann/detail/input/lexer.hpp @@ -908,13 +908,9 @@ class lexer goto scan_number_any1; } - // LCOV_EXCL_START - default: - { - // all other characters are rejected outside scan_number() - assert(false); - } - // LCOV_EXCL_STOP + // all other characters are rejected outside scan_number() + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } scan_number_minus: diff --git a/include/nlohmann/detail/input/parser.hpp b/include/nlohmann/detail/input/parser.hpp index 72895c1d..a3072de4 100644 --- a/include/nlohmann/detail/input/parser.hpp +++ b/include/nlohmann/detail/input/parser.hpp @@ -459,7 +459,7 @@ class parser /// get next token from lexer token_type get_token() { - return (last_token = m_lexer.scan()); + return last_token = m_lexer.scan(); } std::string exception_message(const token_type expected, const std::string& context) diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 3d44e267..154e8b71 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -344,7 +344,7 @@ class json_pointer std::all_of(reference_token.begin(), reference_token.end(), [](const char x) { - return (x >= '0' and x <= '9'); + return x >= '0' and x <= '9'; }); // change value to array for numbers or "-" or to object otherwise @@ -793,7 +793,7 @@ class json_pointer friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { - return (lhs.reference_tokens == rhs.reference_tokens); + return lhs.reference_tokens == rhs.reference_tokens; } friend bool operator!=(json_pointer const& lhs, diff --git a/include/nlohmann/detail/macro_scope.hpp b/include/nlohmann/detail/macro_scope.hpp index 94d081ee..d8408361 100644 --- a/include/nlohmann/detail/macro_scope.hpp +++ b/include/nlohmann/detail/macro_scope.hpp @@ -88,8 +88,8 @@ // manual branch prediction #if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #define JSON_LIKELY(x) __builtin_expect(!!(x), 1) - #define JSON_UNLIKELY(x) __builtin_expect(!!(x), 0) + #define JSON_LIKELY(x) __builtin_expect(static_cast(x), 1) + #define JSON_UNLIKELY(x) __builtin_expect(static_cast(x), 0) #else #define JSON_LIKELY(x) x #define JSON_UNLIKELY(x) x diff --git a/include/nlohmann/detail/output/serializer.hpp b/include/nlohmann/detail/output/serializer.hpp index 31174eac..bb7f1314 100644 --- a/include/nlohmann/detail/output/serializer.hpp +++ b/include/nlohmann/detail/output/serializer.hpp @@ -279,10 +279,8 @@ class serializer return; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } } @@ -480,10 +478,8 @@ class serializer break; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } break; } @@ -545,10 +541,8 @@ class serializer break; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } } } @@ -755,7 +749,7 @@ class serializer std::none_of(number_buffer.begin(), number_buffer.begin() + len + 1, [](char c) { - return (c == '.' or c == 'e'); + return c == '.' or c == 'e'; }); if (value_is_int_like) diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 6bc13727..cbe30fe2 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -1325,7 +1325,7 @@ class basic_json case value_t::discarded: m_type = value_t::discarded; break; - default: + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE } assert_invariant(); @@ -1414,7 +1414,7 @@ class basic_json bool is_an_object = std::all_of(init.begin(), init.end(), [](const detail::json_ref& element_ref) { - return (element_ref->is_array() and element_ref->size() == 2 and (*element_ref)[0].is_string()); + return element_ref->is_array() and element_ref->size() == 2 and (*element_ref)[0].is_string(); }); // adjust type if type deduction is not wanted @@ -2099,7 +2099,7 @@ class basic_json */ constexpr bool is_null() const noexcept { - return (m_type == value_t::null); + return m_type == value_t::null; } /*! @@ -2121,7 +2121,7 @@ class basic_json */ constexpr bool is_boolean() const noexcept { - return (m_type == value_t::boolean); + return m_type == value_t::boolean; } /*! @@ -2180,7 +2180,7 @@ class basic_json */ constexpr bool is_number_integer() const noexcept { - return (m_type == value_t::number_integer or m_type == value_t::number_unsigned); + return m_type == value_t::number_integer or m_type == value_t::number_unsigned; } /*! @@ -2208,7 +2208,7 @@ class basic_json */ constexpr bool is_number_unsigned() const noexcept { - return (m_type == value_t::number_unsigned); + return m_type == value_t::number_unsigned; } /*! @@ -2236,7 +2236,7 @@ class basic_json */ constexpr bool is_number_float() const noexcept { - return (m_type == value_t::number_float); + return m_type == value_t::number_float; } /*! @@ -2258,7 +2258,7 @@ class basic_json */ constexpr bool is_object() const noexcept { - return (m_type == value_t::object); + return m_type == value_t::object; } /*! @@ -2280,7 +2280,7 @@ class basic_json */ constexpr bool is_array() const noexcept { - return (m_type == value_t::array); + return m_type == value_t::array; } /*! @@ -2302,7 +2302,7 @@ class basic_json */ constexpr bool is_string() const noexcept { - return (m_type == value_t::string); + return m_type == value_t::string; } /*! @@ -2329,7 +2329,7 @@ class basic_json */ constexpr bool is_discarded() const noexcept { - return (m_type == value_t::discarded); + return m_type == value_t::discarded; } /*! @@ -3990,7 +3990,7 @@ class basic_json template bool contains(KeyT&& key) const { - return (is_object() and m_value.object->find(std::forward(key)) != m_value.object->end()); + return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); } /// @} @@ -5569,28 +5569,28 @@ class basic_json switch (lhs_type) { case value_t::array: - return (*lhs.m_value.array == *rhs.m_value.array); + return *lhs.m_value.array == *rhs.m_value.array; case value_t::object: - return (*lhs.m_value.object == *rhs.m_value.object); + return *lhs.m_value.object == *rhs.m_value.object; case value_t::null: return true; case value_t::string: - return (*lhs.m_value.string == *rhs.m_value.string); + return *lhs.m_value.string == *rhs.m_value.string; case value_t::boolean: - return (lhs.m_value.boolean == rhs.m_value.boolean); + return lhs.m_value.boolean == rhs.m_value.boolean; case value_t::number_integer: - return (lhs.m_value.number_integer == rhs.m_value.number_integer); + return lhs.m_value.number_integer == rhs.m_value.number_integer; case value_t::number_unsigned: - return (lhs.m_value.number_unsigned == rhs.m_value.number_unsigned); + return lhs.m_value.number_unsigned == rhs.m_value.number_unsigned; case value_t::number_float: - return (lhs.m_value.number_float == rhs.m_value.number_float); + return lhs.m_value.number_float == rhs.m_value.number_float; default: return false; @@ -5598,27 +5598,27 @@ class basic_json } else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_float) { - return (static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float); + return static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float; } else if (lhs_type == value_t::number_float and rhs_type == value_t::number_integer) { - return (lhs.m_value.number_float == static_cast(rhs.m_value.number_integer)); + return lhs.m_value.number_float == static_cast(rhs.m_value.number_integer); } else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_float) { - return (static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float); + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float; } else if (lhs_type == value_t::number_float and rhs_type == value_t::number_unsigned) { - return (lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned)); + return lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned); } else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_integer) { - return (static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer); + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer; } else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_unsigned) { - return (lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned)); + return lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned); } return false; @@ -5632,7 +5632,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator==(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs == basic_json(rhs)); + return lhs == basic_json(rhs); } /*! @@ -5643,7 +5643,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator==(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) == rhs); + return basic_json(lhs) == rhs; } /*! @@ -5677,7 +5677,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator!=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs != basic_json(rhs)); + return lhs != basic_json(rhs); } /*! @@ -5688,7 +5688,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator!=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) != rhs); + return basic_json(lhs) != rhs; } /*! @@ -5727,7 +5727,7 @@ class basic_json switch (lhs_type) { case value_t::array: - return (*lhs.m_value.array) < (*rhs.m_value.array); + return *lhs.m_value.array < *rhs.m_value.array; case value_t::object: return *lhs.m_value.object < *rhs.m_value.object; @@ -5793,7 +5793,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs < basic_json(rhs)); + return lhs < basic_json(rhs); } /*! @@ -5804,7 +5804,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) < rhs); + return basic_json(lhs) < rhs; } /*! @@ -5839,7 +5839,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs <= basic_json(rhs)); + return lhs <= basic_json(rhs); } /*! @@ -5850,7 +5850,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) <= rhs); + return basic_json(lhs) <= rhs; } /*! @@ -5885,7 +5885,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs > basic_json(rhs)); + return lhs > basic_json(rhs); } /*! @@ -5896,7 +5896,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) > rhs); + return basic_json(lhs) > rhs; } /*! @@ -5931,7 +5931,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs >= basic_json(rhs)); + return lhs >= basic_json(rhs); } /*! @@ -5942,7 +5942,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) >= rhs); + return basic_json(lhs) >= rhs; } /// @} @@ -5988,8 +5988,8 @@ class basic_json friend std::ostream& operator<<(std::ostream& o, const basic_json& j) { // read width member and use it as indentation parameter if nonzero - const bool pretty_print = (o.width() > 0); - const auto indentation = (pretty_print ? o.width() : 0); + const bool pretty_print = o.width() > 0; + const auto indentation = pretty_print ? o.width() : 0; // reset width to 0 for subsequent calls to this stream o.width(0); @@ -7495,60 +7495,55 @@ class basic_json if (ptr.empty()) { result = val; + return; } - else + + // make sure the top element of the pointer exists + json_pointer top_pointer = ptr.top(); + if (top_pointer != ptr) { - // make sure the top element of the pointer exists - json_pointer top_pointer = ptr.top(); - if (top_pointer != ptr) + result.at(top_pointer); + } + + // get reference to parent of JSON pointer ptr + const auto last_path = ptr.pop_back(); + basic_json& parent = result[ptr]; + + switch (parent.m_type) + { + case value_t::null: + case value_t::object: { - result.at(top_pointer); + // use operator[] to add value + parent[last_path] = val; + break; } - // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); - basic_json& parent = result[ptr]; - - switch (parent.m_type) + case value_t::array: { - case value_t::null: - case value_t::object: + if (last_path == "-") { - // use operator[] to add value - parent[last_path] = val; - break; + // special case: append to back + parent.push_back(val); } - - case value_t::array: + else { - if (last_path == "-") + const auto idx = json_pointer::array_index(last_path); + if (JSON_UNLIKELY(static_cast(idx) > parent.size())) { - // special case: append to back - parent.push_back(val); + // avoid undefined behavior + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } - else - { - const auto idx = json_pointer::array_index(last_path); - if (JSON_UNLIKELY(static_cast(idx) > parent.size())) - { - // avoid undefined behavior - JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); - } - // default case: insert add offset - parent.insert(parent.begin() + static_cast(idx), val); - } - break; + // default case: insert add offset + parent.insert(parent.begin() + static_cast(idx), val); } - - // LCOV_EXCL_START - default: - { - // if there exists a parent it cannot be primitive - assert(false); - } - // LCOV_EXCL_STOP + break; } + + // if there exists a parent it cannot be primitive + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } }; @@ -7768,106 +7763,105 @@ class basic_json { {"op", "replace"}, {"path", path}, {"value", target} }); + return result; } - else + + switch (source.type()) { - switch (source.type()) + case value_t::array: { - case value_t::array: + // first pass: traverse common elements + std::size_t i = 0; + while (i < source.size() and i < target.size()) { - // first pass: traverse common elements - std::size_t i = 0; - while (i < source.size() and i < target.size()) - { - // recursive call to compare array values at index i - auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - ++i; - } - - // i now reached the end of at least one array - // in a second pass, traverse the remaining elements - - // remove my remaining elements - const auto end_index = static_cast(result.size()); - while (i < source.size()) - { - // add operations in reverse order to avoid invalid - // indices - result.insert(result.begin() + end_index, object( - { - {"op", "remove"}, - {"path", path + "/" + std::to_string(i)} - })); - ++i; - } - - // add other remaining elements - while (i < target.size()) - { - result.push_back( - { - {"op", "add"}, - {"path", path + "/" + std::to_string(i)}, - {"value", target[i]} - }); - ++i; - } - - break; + // recursive call to compare array values at index i + auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + ++i; } - case value_t::object: + // i now reached the end of at least one array + // in a second pass, traverse the remaining elements + + // remove my remaining elements + const auto end_index = static_cast(result.size()); + while (i < source.size()) { - // first pass: traverse this object's elements - for (auto it = source.cbegin(); it != source.cend(); ++it) + // add operations in reverse order to avoid invalid + // indices + result.insert(result.begin() + end_index, object( { - // escape the key name to be used in a JSON patch - const auto key = json_pointer::escape(it.key()); - - if (target.find(it.key()) != target.end()) - { - // recursive call to compare object values at key it - auto temp_diff = diff(it.value(), target[it.key()], path + "/" + key); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - } - else - { - // found a key that is not in o -> remove it - result.push_back(object( - { - {"op", "remove"}, {"path", path + "/" + key} - })); - } - } - - // second pass: traverse other object's elements - for (auto it = target.cbegin(); it != target.cend(); ++it) - { - if (source.find(it.key()) == source.end()) - { - // found a key that is not in this -> add it - const auto key = json_pointer::escape(it.key()); - result.push_back( - { - {"op", "add"}, {"path", path + "/" + key}, - {"value", it.value()} - }); - } - } - - break; + {"op", "remove"}, + {"path", path + "/" + std::to_string(i)} + })); + ++i; } - default: + // add other remaining elements + while (i < target.size()) { - // both primitive type: replace value result.push_back( { - {"op", "replace"}, {"path", path}, {"value", target} + {"op", "add"}, + {"path", path + "/" + std::to_string(i)}, + {"value", target[i]} }); - break; + ++i; } + + break; + } + + case value_t::object: + { + // first pass: traverse this object's elements + for (auto it = source.cbegin(); it != source.cend(); ++it) + { + // escape the key name to be used in a JSON patch + const auto key = json_pointer::escape(it.key()); + + if (target.find(it.key()) != target.end()) + { + // recursive call to compare object values at key it + auto temp_diff = diff(it.value(), target[it.key()], path + "/" + key); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + } + else + { + // found a key that is not in o -> remove it + result.push_back(object( + { + {"op", "remove"}, {"path", path + "/" + key} + })); + } + } + + // second pass: traverse other object's elements + for (auto it = target.cbegin(); it != target.cend(); ++it) + { + if (source.find(it.key()) == source.end()) + { + // found a key that is not in this -> add it + const auto key = json_pointer::escape(it.key()); + result.push_back( + { + {"op", "add"}, {"path", path + "/" + key}, + {"value", it.value()} + }); + } + } + + break; + } + + default: + { + // both primitive type: replace value + result.push_back( + { + {"op", "replace"}, {"path", path}, {"value", target} + }); + break; } } diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index f5b3aa7c..010ac75e 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -539,8 +539,8 @@ class other_error : public exception // manual branch prediction #if defined(__clang__) || defined(__GNUC__) || defined(__GNUG__) - #define JSON_LIKELY(x) __builtin_expect(!!(x), 1) - #define JSON_UNLIKELY(x) __builtin_expect(!!(x), 0) + #define JSON_LIKELY(x) __builtin_expect(static_cast(x), 1) + #define JSON_UNLIKELY(x) __builtin_expect(static_cast(x), 0) #else #define JSON_LIKELY(x) x #define JSON_UNLIKELY(x) x @@ -2521,7 +2521,7 @@ template class wide_string_input_adapter : public input_adapter_protocol { public: - explicit wide_string_input_adapter(const WideStringType& w) noexcept + explicit wide_string_input_adapter(const WideStringType& w) noexcept : str(w) {} @@ -2981,12 +2981,11 @@ class json_sax_dom_parser ref_stack.back()->m_value.array->emplace_back(std::forward(v)); return &(ref_stack.back()->m_value.array->back()); } - else - { - assert(object_element); - *object_element = BasicJsonType(std::forward(v)); - return object_element; - } + + assert(ref_stack.back()->is_object()) + assert(object_element); + *object_element = BasicJsonType(std::forward(v)); + return object_element; } /// the parsed JSON value @@ -3073,13 +3072,9 @@ class json_sax_dom_callback_parser ref_stack.push_back(val.second); // check object limit - if (ref_stack.back()) + if (ref_stack.back() and JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) { - if (JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) - { - JSON_THROW(out_of_range::create(408, - "excessive object size: " + std::to_string(len))); - } + JSON_THROW(out_of_range::create(408, "excessive object size: " + std::to_string(len))); } return true; @@ -3104,13 +3099,10 @@ class json_sax_dom_callback_parser bool end_object() { - if (ref_stack.back()) + if (ref_stack.back() and not callback(static_cast(ref_stack.size()) - 1, parse_event_t::object_end, *ref_stack.back())) { - if (not callback(static_cast(ref_stack.size()) - 1, parse_event_t::object_end, *ref_stack.back())) - { - // discard object - *ref_stack.back() = discarded; - } + // discard object + *ref_stack.back() = discarded; } assert(not ref_stack.empty()); @@ -3118,18 +3110,15 @@ class json_sax_dom_callback_parser ref_stack.pop_back(); keep_stack.pop_back(); - if (not ref_stack.empty() and ref_stack.back()) + if (not ref_stack.empty() and ref_stack.back() and ref_stack.back()->is_object()) { // remove discarded value - if (ref_stack.back()->is_object()) + for (auto it = ref_stack.back()->begin(); it != ref_stack.back()->end(); ++it) { - for (auto it = ref_stack.back()->begin(); it != ref_stack.back()->end(); ++it) + if (it->is_discarded()) { - if (it->is_discarded()) - { - ref_stack.back()->erase(it); - break; - } + ref_stack.back()->erase(it); + break; } } } @@ -3146,13 +3135,9 @@ class json_sax_dom_callback_parser ref_stack.push_back(val.second); // check array limit - if (ref_stack.back()) + if (ref_stack.back() and JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) { - if (JSON_UNLIKELY(len != std::size_t(-1) and len > ref_stack.back()->max_size())) - { - JSON_THROW(out_of_range::create(408, - "excessive array size: " + std::to_string(len))); - } + JSON_THROW(out_of_range::create(408, "excessive array size: " + std::to_string(len))); } return true; @@ -3178,12 +3163,9 @@ class json_sax_dom_callback_parser keep_stack.pop_back(); // remove discarded value - if (not keep and not ref_stack.empty()) + if (not keep and not ref_stack.empty() and ref_stack.back()->is_array()) { - if (ref_stack.back()->is_array()) - { - ref_stack.back()->m_value.array->pop_back(); - } + ref_stack.back()->m_value.array->pop_back(); } return true; @@ -3278,27 +3260,28 @@ class json_sax_dom_callback_parser // we now only expect arrays and objects assert(ref_stack.back()->is_array() or ref_stack.back()->is_object()); + // array if (ref_stack.back()->is_array()) { ref_stack.back()->m_value.array->push_back(std::move(value)); return {true, &(ref_stack.back()->m_value.array->back())}; } - else + + // object + assert(ref_stack.back()->is_object()) + // check if we should store an element for the current key + assert(not key_keep_stack.empty()); + const bool store_element = key_keep_stack.back(); + key_keep_stack.pop_back(); + + if (not store_element) { - // check if we should store an element for the current key - assert(not key_keep_stack.empty()); - const bool store_element = key_keep_stack.back(); - key_keep_stack.pop_back(); - - if (not store_element) - { - return {false, nullptr}; - } - - assert(object_element); - *object_element = std::move(value); - return {true, object_element}; + return {false, nullptr}; } + + assert(object_element); + *object_element = std::move(value); + return {true, object_element}; } /// the parsed JSON value @@ -3616,10 +3599,8 @@ class binary_reader result = parse_ubjson_internal(); break; - // LCOV_EXCL_START - default: - assert(false); - // LCOV_EXCL_STOP + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } // strict mode: next byte must be EOF @@ -3653,7 +3634,7 @@ class binary_reader */ static constexpr bool little_endianess(int num = 1) noexcept { - return (*reinterpret_cast(&num) == 1); + return *reinterpret_cast(&num) == 1; } private: @@ -3830,12 +3811,9 @@ class binary_reader return false; } - if (not is_array) + if (not is_array and not sax->key(key)) { - if (not sax->key(key)) - { - return false; - } + return false; } if (JSON_UNLIKELY(not parse_bson_element_internal(element_type, element_type_parse_position))) @@ -5343,7 +5321,7 @@ class binary_reader int get() { ++chars_read; - return (current = ia->get_character()); + return current = ia->get_character(); } /*! @@ -5489,10 +5467,8 @@ class binary_reader error_msg += "BSON"; break; - // LCOV_EXCL_START - default: - assert(false); - // LCOV_EXCL_STOP + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } return error_msg + " " + context + ": " + detail; @@ -6433,13 +6409,9 @@ class lexer goto scan_number_any1; } - // LCOV_EXCL_START - default: - { - // all other characters are rejected outside scan_number() - assert(false); - } - // LCOV_EXCL_STOP + // all other characters are rejected outside scan_number() + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } scan_number_minus: @@ -7501,7 +7473,7 @@ class parser /// get next token from lexer token_type get_token() { - return (last_token = m_lexer.scan()); + return last_token = m_lexer.scan(); } std::string exception_message(const token_type expected, const std::string& context) @@ -8791,7 +8763,7 @@ class json_pointer std::all_of(reference_token.begin(), reference_token.end(), [](const char x) { - return (x >= '0' and x <= '9'); + return x >= '0' and x <= '9'; }); // change value to array for numbers or "-" or to object otherwise @@ -9240,7 +9212,7 @@ class json_pointer friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { - return (lhs.reference_tokens == rhs.reference_tokens); + return lhs.reference_tokens == rhs.reference_tokens; } friend bool operator!=(json_pointer const& lhs, @@ -11016,7 +10988,7 @@ boundaries compute_boundaries(FloatType value) const std::uint64_t E = bits >> (kPrecision - 1); const std::uint64_t F = bits & (kHiddenBit - 1); - const bool is_denormal = (E == 0); + const bool is_denormal = E == 0; const diyfp v = is_denormal ? diyfp(F, kMinExp) : diyfp(F + kHiddenBit, static_cast(E) - kBias); @@ -11042,7 +11014,7 @@ boundaries compute_boundaries(FloatType value) // -----------------+------+------+-------------+-------------+--- (B) // v- m- v m+ v+ - const bool lower_boundary_is_closer = (F == 0 and E > 1); + const bool lower_boundary_is_closer = F == 0 and E > 1; const diyfp m_plus = diyfp(2 * v.f + 1, v.e - 1); const diyfp m_minus = lower_boundary_is_closer ? diyfp(4 * v.f - 1, v.e - 2) // (B) @@ -12178,10 +12150,8 @@ class serializer return; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } } @@ -12379,10 +12349,8 @@ class serializer break; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } break; } @@ -12444,10 +12412,8 @@ class serializer break; } - default: - { + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE - } } } } @@ -12654,7 +12620,7 @@ class serializer std::none_of(number_buffer.begin(), number_buffer.begin() + len + 1, [](char c) { - return (c == '.' or c == 'e'); + return c == '.' or c == 'e'; }); if (value_is_int_like) @@ -14002,7 +13968,7 @@ class basic_json case value_t::discarded: m_type = value_t::discarded; break; - default: + default: // LCOV_EXCL_LINE assert(false); // LCOV_EXCL_LINE } assert_invariant(); @@ -14091,7 +14057,7 @@ class basic_json bool is_an_object = std::all_of(init.begin(), init.end(), [](const detail::json_ref& element_ref) { - return (element_ref->is_array() and element_ref->size() == 2 and (*element_ref)[0].is_string()); + return element_ref->is_array() and element_ref->size() == 2 and (*element_ref)[0].is_string(); }); // adjust type if type deduction is not wanted @@ -14776,7 +14742,7 @@ class basic_json */ constexpr bool is_null() const noexcept { - return (m_type == value_t::null); + return m_type == value_t::null; } /*! @@ -14798,7 +14764,7 @@ class basic_json */ constexpr bool is_boolean() const noexcept { - return (m_type == value_t::boolean); + return m_type == value_t::boolean; } /*! @@ -14857,7 +14823,7 @@ class basic_json */ constexpr bool is_number_integer() const noexcept { - return (m_type == value_t::number_integer or m_type == value_t::number_unsigned); + return m_type == value_t::number_integer or m_type == value_t::number_unsigned; } /*! @@ -14885,7 +14851,7 @@ class basic_json */ constexpr bool is_number_unsigned() const noexcept { - return (m_type == value_t::number_unsigned); + return m_type == value_t::number_unsigned; } /*! @@ -14913,7 +14879,7 @@ class basic_json */ constexpr bool is_number_float() const noexcept { - return (m_type == value_t::number_float); + return m_type == value_t::number_float; } /*! @@ -14935,7 +14901,7 @@ class basic_json */ constexpr bool is_object() const noexcept { - return (m_type == value_t::object); + return m_type == value_t::object; } /*! @@ -14957,7 +14923,7 @@ class basic_json */ constexpr bool is_array() const noexcept { - return (m_type == value_t::array); + return m_type == value_t::array; } /*! @@ -14979,7 +14945,7 @@ class basic_json */ constexpr bool is_string() const noexcept { - return (m_type == value_t::string); + return m_type == value_t::string; } /*! @@ -15006,7 +14972,7 @@ class basic_json */ constexpr bool is_discarded() const noexcept { - return (m_type == value_t::discarded); + return m_type == value_t::discarded; } /*! @@ -16667,7 +16633,7 @@ class basic_json template bool contains(KeyT&& key) const { - return (is_object() and m_value.object->find(std::forward(key)) != m_value.object->end()); + return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); } /// @} @@ -18246,28 +18212,28 @@ class basic_json switch (lhs_type) { case value_t::array: - return (*lhs.m_value.array == *rhs.m_value.array); + return *lhs.m_value.array == *rhs.m_value.array; case value_t::object: - return (*lhs.m_value.object == *rhs.m_value.object); + return *lhs.m_value.object == *rhs.m_value.object; case value_t::null: return true; case value_t::string: - return (*lhs.m_value.string == *rhs.m_value.string); + return *lhs.m_value.string == *rhs.m_value.string; case value_t::boolean: - return (lhs.m_value.boolean == rhs.m_value.boolean); + return lhs.m_value.boolean == rhs.m_value.boolean; case value_t::number_integer: - return (lhs.m_value.number_integer == rhs.m_value.number_integer); + return lhs.m_value.number_integer == rhs.m_value.number_integer; case value_t::number_unsigned: - return (lhs.m_value.number_unsigned == rhs.m_value.number_unsigned); + return lhs.m_value.number_unsigned == rhs.m_value.number_unsigned; case value_t::number_float: - return (lhs.m_value.number_float == rhs.m_value.number_float); + return lhs.m_value.number_float == rhs.m_value.number_float; default: return false; @@ -18275,27 +18241,27 @@ class basic_json } else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_float) { - return (static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float); + return static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float; } else if (lhs_type == value_t::number_float and rhs_type == value_t::number_integer) { - return (lhs.m_value.number_float == static_cast(rhs.m_value.number_integer)); + return lhs.m_value.number_float == static_cast(rhs.m_value.number_integer); } else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_float) { - return (static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float); + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float; } else if (lhs_type == value_t::number_float and rhs_type == value_t::number_unsigned) { - return (lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned)); + return lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned); } else if (lhs_type == value_t::number_unsigned and rhs_type == value_t::number_integer) { - return (static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer); + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer; } else if (lhs_type == value_t::number_integer and rhs_type == value_t::number_unsigned) { - return (lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned)); + return lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned); } return false; @@ -18309,7 +18275,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator==(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs == basic_json(rhs)); + return lhs == basic_json(rhs); } /*! @@ -18320,7 +18286,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator==(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) == rhs); + return basic_json(lhs) == rhs; } /*! @@ -18354,7 +18320,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator!=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs != basic_json(rhs)); + return lhs != basic_json(rhs); } /*! @@ -18365,7 +18331,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator!=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) != rhs); + return basic_json(lhs) != rhs; } /*! @@ -18404,7 +18370,7 @@ class basic_json switch (lhs_type) { case value_t::array: - return (*lhs.m_value.array) < (*rhs.m_value.array); + return *lhs.m_value.array < *rhs.m_value.array; case value_t::object: return *lhs.m_value.object < *rhs.m_value.object; @@ -18470,7 +18436,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs < basic_json(rhs)); + return lhs < basic_json(rhs); } /*! @@ -18481,7 +18447,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) < rhs); + return basic_json(lhs) < rhs; } /*! @@ -18516,7 +18482,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs <= basic_json(rhs)); + return lhs <= basic_json(rhs); } /*! @@ -18527,7 +18493,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator<=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) <= rhs); + return basic_json(lhs) <= rhs; } /*! @@ -18562,7 +18528,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs > basic_json(rhs)); + return lhs > basic_json(rhs); } /*! @@ -18573,7 +18539,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) > rhs); + return basic_json(lhs) > rhs; } /*! @@ -18608,7 +18574,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>=(const_reference lhs, const ScalarType rhs) noexcept { - return (lhs >= basic_json(rhs)); + return lhs >= basic_json(rhs); } /*! @@ -18619,7 +18585,7 @@ class basic_json std::is_scalar::value, int>::type = 0> friend bool operator>=(const ScalarType lhs, const_reference rhs) noexcept { - return (basic_json(lhs) >= rhs); + return basic_json(lhs) >= rhs; } /// @} @@ -18665,8 +18631,8 @@ class basic_json friend std::ostream& operator<<(std::ostream& o, const basic_json& j) { // read width member and use it as indentation parameter if nonzero - const bool pretty_print = (o.width() > 0); - const auto indentation = (pretty_print ? o.width() : 0); + const bool pretty_print = o.width() > 0; + const auto indentation = pretty_print ? o.width() : 0; // reset width to 0 for subsequent calls to this stream o.width(0); @@ -20172,60 +20138,55 @@ class basic_json if (ptr.empty()) { result = val; + return; } - else + + // make sure the top element of the pointer exists + json_pointer top_pointer = ptr.top(); + if (top_pointer != ptr) { - // make sure the top element of the pointer exists - json_pointer top_pointer = ptr.top(); - if (top_pointer != ptr) + result.at(top_pointer); + } + + // get reference to parent of JSON pointer ptr + const auto last_path = ptr.pop_back(); + basic_json& parent = result[ptr]; + + switch (parent.m_type) + { + case value_t::null: + case value_t::object: { - result.at(top_pointer); + // use operator[] to add value + parent[last_path] = val; + break; } - // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); - basic_json& parent = result[ptr]; - - switch (parent.m_type) + case value_t::array: { - case value_t::null: - case value_t::object: + if (last_path == "-") { - // use operator[] to add value - parent[last_path] = val; - break; + // special case: append to back + parent.push_back(val); } - - case value_t::array: + else { - if (last_path == "-") + const auto idx = json_pointer::array_index(last_path); + if (JSON_UNLIKELY(static_cast(idx) > parent.size())) { - // special case: append to back - parent.push_back(val); + // avoid undefined behavior + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); } - else - { - const auto idx = json_pointer::array_index(last_path); - if (JSON_UNLIKELY(static_cast(idx) > parent.size())) - { - // avoid undefined behavior - JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range")); - } - // default case: insert add offset - parent.insert(parent.begin() + static_cast(idx), val); - } - break; + // default case: insert add offset + parent.insert(parent.begin() + static_cast(idx), val); } - - // LCOV_EXCL_START - default: - { - // if there exists a parent it cannot be primitive - assert(false); - } - // LCOV_EXCL_STOP + break; } + + // if there exists a parent it cannot be primitive + default: // LCOV_EXCL_LINE + assert(false); // LCOV_EXCL_LINE } }; @@ -20445,106 +20406,105 @@ class basic_json { {"op", "replace"}, {"path", path}, {"value", target} }); + return result; } - else + + switch (source.type()) { - switch (source.type()) + case value_t::array: { - case value_t::array: + // first pass: traverse common elements + std::size_t i = 0; + while (i < source.size() and i < target.size()) { - // first pass: traverse common elements - std::size_t i = 0; - while (i < source.size() and i < target.size()) - { - // recursive call to compare array values at index i - auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - ++i; - } - - // i now reached the end of at least one array - // in a second pass, traverse the remaining elements - - // remove my remaining elements - const auto end_index = static_cast(result.size()); - while (i < source.size()) - { - // add operations in reverse order to avoid invalid - // indices - result.insert(result.begin() + end_index, object( - { - {"op", "remove"}, - {"path", path + "/" + std::to_string(i)} - })); - ++i; - } - - // add other remaining elements - while (i < target.size()) - { - result.push_back( - { - {"op", "add"}, - {"path", path + "/" + std::to_string(i)}, - {"value", target[i]} - }); - ++i; - } - - break; + // recursive call to compare array values at index i + auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + ++i; } - case value_t::object: + // i now reached the end of at least one array + // in a second pass, traverse the remaining elements + + // remove my remaining elements + const auto end_index = static_cast(result.size()); + while (i < source.size()) { - // first pass: traverse this object's elements - for (auto it = source.cbegin(); it != source.cend(); ++it) + // add operations in reverse order to avoid invalid + // indices + result.insert(result.begin() + end_index, object( { - // escape the key name to be used in a JSON patch - const auto key = json_pointer::escape(it.key()); - - if (target.find(it.key()) != target.end()) - { - // recursive call to compare object values at key it - auto temp_diff = diff(it.value(), target[it.key()], path + "/" + key); - result.insert(result.end(), temp_diff.begin(), temp_diff.end()); - } - else - { - // found a key that is not in o -> remove it - result.push_back(object( - { - {"op", "remove"}, {"path", path + "/" + key} - })); - } - } - - // second pass: traverse other object's elements - for (auto it = target.cbegin(); it != target.cend(); ++it) - { - if (source.find(it.key()) == source.end()) - { - // found a key that is not in this -> add it - const auto key = json_pointer::escape(it.key()); - result.push_back( - { - {"op", "add"}, {"path", path + "/" + key}, - {"value", it.value()} - }); - } - } - - break; + {"op", "remove"}, + {"path", path + "/" + std::to_string(i)} + })); + ++i; } - default: + // add other remaining elements + while (i < target.size()) { - // both primitive type: replace value result.push_back( { - {"op", "replace"}, {"path", path}, {"value", target} + {"op", "add"}, + {"path", path + "/" + std::to_string(i)}, + {"value", target[i]} }); - break; + ++i; } + + break; + } + + case value_t::object: + { + // first pass: traverse this object's elements + for (auto it = source.cbegin(); it != source.cend(); ++it) + { + // escape the key name to be used in a JSON patch + const auto key = json_pointer::escape(it.key()); + + if (target.find(it.key()) != target.end()) + { + // recursive call to compare object values at key it + auto temp_diff = diff(it.value(), target[it.key()], path + "/" + key); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + } + else + { + // found a key that is not in o -> remove it + result.push_back(object( + { + {"op", "remove"}, {"path", path + "/" + key} + })); + } + } + + // second pass: traverse other object's elements + for (auto it = target.cbegin(); it != target.cend(); ++it) + { + if (source.find(it.key()) == source.end()) + { + // found a key that is not in this -> add it + const auto key = json_pointer::escape(it.key()); + result.push_back( + { + {"op", "add"}, {"path", path + "/" + key}, + {"value", it.value()} + }); + } + } + + break; + } + + default: + { + // both primitive type: replace value + result.push_back( + { + {"op", "replace"}, {"path", path}, {"value", target} + }); + break; } } From 155d196bfab4416c5c7431b96644df7311576f55 Mon Sep 17 00:00:00 2001 From: Viktor Kirilov Date: Mon, 18 Mar 2019 17:58:35 +0200 Subject: [PATCH 044/220] Update CMakeLists.txt --- test/CMakeLists.txt | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 1ac271c1..b2f59d5f 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -65,11 +65,7 @@ set_target_properties(doctest_main PROPERTIES COMPILE_OPTIONS "$<$:/EHsc;$<$:/Od>>" ) -if (${CMAKE_VERSION} VERSION_LESS "3.8.0") - target_compile_features(catch_main PUBLIC cxx_range_for) -else() - target_compile_features(catch_main PUBLIC cxx_std_11) -endif() +target_compile_features(doctest_main PUBLIC cxx_std_11) target_include_directories(doctest_main PRIVATE "thirdparty/doctest") # https://stackoverflow.com/questions/2368811/how-to-set-warning-level-in-cmake From 37a72dac48035e12df13c923e19f10cbe8a22f5f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Mon, 18 Mar 2019 17:23:34 +0100 Subject: [PATCH 045/220] :green_heart: forgot two semicolons --- include/nlohmann/detail/input/json_sax.hpp | 4 ++-- single_include/nlohmann/json.hpp | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/include/nlohmann/detail/input/json_sax.hpp b/include/nlohmann/detail/input/json_sax.hpp index fd1ba76c..66651a90 100644 --- a/include/nlohmann/detail/input/json_sax.hpp +++ b/include/nlohmann/detail/input/json_sax.hpp @@ -304,7 +304,7 @@ class json_sax_dom_parser return &(ref_stack.back()->m_value.array->back()); } - assert(ref_stack.back()->is_object()) + assert(ref_stack.back()->is_object()); assert(object_element); *object_element = BasicJsonType(std::forward(v)); return object_element; @@ -590,7 +590,7 @@ class json_sax_dom_callback_parser } // object - assert(ref_stack.back()->is_object()) + assert(ref_stack.back()->is_object()); // check if we should store an element for the current key assert(not key_keep_stack.empty()); const bool store_element = key_keep_stack.back(); diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 010ac75e..92293fb0 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -2982,7 +2982,7 @@ class json_sax_dom_parser return &(ref_stack.back()->m_value.array->back()); } - assert(ref_stack.back()->is_object()) + assert(ref_stack.back()->is_object()); assert(object_element); *object_element = BasicJsonType(std::forward(v)); return object_element; @@ -3268,7 +3268,7 @@ class json_sax_dom_callback_parser } // object - assert(ref_stack.back()->is_object()) + assert(ref_stack.back()->is_object()); // check if we should store an element for the current key assert(not key_keep_stack.empty()); const bool store_element = key_keep_stack.back(); From b224c52376110fc9c5aa1a86e3197be887a00a66 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 19 Mar 2019 09:17:14 +0100 Subject: [PATCH 046/220] :art: cleanup --- Makefile | 2 +- include/nlohmann/detail/input/parser.hpp | 157 +++++++++++----------- include/nlohmann/json.hpp | 3 +- single_include/nlohmann/json.hpp | 160 +++++++++++------------ 4 files changed, 153 insertions(+), 169 deletions(-) diff --git a/Makefile b/Makefile index 086b07f1..24c77f51 100644 --- a/Makefile +++ b/Makefile @@ -442,7 +442,7 @@ fuzzing-stop: # call cppcheck on the main header file cppcheck: - cppcheck --enable=warning --inconclusive --force --std=c++11 $(SRCS) --error-exitcode=1 + cppcheck --enable=warning --inline-suppr --inconclusive --force --std=c++11 $(SRCS) --error-exitcode=1 # compile and check with Clang Static Analyzer clang_analyze: diff --git a/include/nlohmann/detail/input/parser.hpp b/include/nlohmann/detail/input/parser.hpp index a3072de4..6c03181e 100644 --- a/include/nlohmann/detail/input/parser.hpp +++ b/include/nlohmann/detail/input/parser.hpp @@ -262,14 +262,13 @@ class parser m_lexer.get_token_string(), out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); } - else + + if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) { - if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) - { - return false; - } - break; + return false; } + + break; } case token_type::literal_false: @@ -355,103 +354,95 @@ class parser // empty stack: we reached the end of the hierarchy: done return true; } - else + + if (states.back()) // array { - if (states.back()) // array + // comma -> next value + if (get_token() == token_type::value_separator) { - // comma -> next value - if (get_token() == token_type::value_separator) + // parse a new value + get_token(); + continue; + } + + // closing ] + if (JSON_LIKELY(last_token == token_type::end_array)) + { + if (JSON_UNLIKELY(not sax->end_array())) { - // parse a new value - get_token(); - continue; + return false; } - // closing ] - if (JSON_LIKELY(last_token == token_type::end_array)) - { - if (JSON_UNLIKELY(not sax->end_array())) - { - return false; - } + // We are done with this array. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } - // We are done with this array. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_array, "array"))); + } + else // object + { + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse key + if (JSON_UNLIKELY(get_token() != token_type::value_string)) { return sax->parse_error(m_lexer.get_position(), m_lexer.get_token_string(), parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_array, "array"))); + exception_message(token_type::value_string, "object key"))); } - } - else // object - { - // comma -> next value - if (get_token() == token_type::value_separator) + + if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) { - // parse key - if (JSON_UNLIKELY(get_token() != token_type::value_string)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::value_string, "object key"))); - } - else - { - if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) - { - return false; - } - } - - // parse separator (:) - if (JSON_UNLIKELY(get_token() != token_type::name_separator)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::name_separator, "object separator"))); - } - - // parse values - get_token(); - continue; + return false; } - // closing } - if (JSON_LIKELY(last_token == token_type::end_object)) - { - if (JSON_UNLIKELY(not sax->end_object())) - { - return false; - } - - // We are done with this object. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else + // parse separator (:) + if (JSON_UNLIKELY(get_token() != token_type::name_separator)) { return sax->parse_error(m_lexer.get_position(), m_lexer.get_token_string(), parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_object, "object"))); + exception_message(token_type::name_separator, "object separator"))); } + + // parse values + get_token(); + continue; } + + // closing } + if (JSON_LIKELY(last_token == token_type::end_object)) + { + if (JSON_UNLIKELY(not sax->end_object())) + { + return false; + } + + // We are done with this object. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_object, "object"))); } } } diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index cbe30fe2..6a5c1075 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -4777,7 +4777,8 @@ class basic_json // add element to array (move semantics) m_value.array->push_back(std::move(val)); - // invalidate object + // invalidate object: mark it null so we do not call the destructor + // cppcheck-suppress accessMoved val.m_type = value_t::null; } diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 92293fb0..7509df49 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -7276,14 +7276,13 @@ class parser m_lexer.get_token_string(), out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'")); } - else + + if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) { - if (JSON_UNLIKELY(not sax->number_float(res, m_lexer.get_string()))) - { - return false; - } - break; + return false; } + + break; } case token_type::literal_false: @@ -7369,103 +7368,95 @@ class parser // empty stack: we reached the end of the hierarchy: done return true; } - else + + if (states.back()) // array { - if (states.back()) // array + // comma -> next value + if (get_token() == token_type::value_separator) { - // comma -> next value - if (get_token() == token_type::value_separator) + // parse a new value + get_token(); + continue; + } + + // closing ] + if (JSON_LIKELY(last_token == token_type::end_array)) + { + if (JSON_UNLIKELY(not sax->end_array())) { - // parse a new value - get_token(); - continue; + return false; } - // closing ] - if (JSON_LIKELY(last_token == token_type::end_array)) - { - if (JSON_UNLIKELY(not sax->end_array())) - { - return false; - } + // We are done with this array. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } - // We are done with this array. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_array, "array"))); + } + else // object + { + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse key + if (JSON_UNLIKELY(get_token() != token_type::value_string)) { return sax->parse_error(m_lexer.get_position(), m_lexer.get_token_string(), parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_array, "array"))); + exception_message(token_type::value_string, "object key"))); } - } - else // object - { - // comma -> next value - if (get_token() == token_type::value_separator) + + if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) { - // parse key - if (JSON_UNLIKELY(get_token() != token_type::value_string)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::value_string, "object key"))); - } - else - { - if (JSON_UNLIKELY(not sax->key(m_lexer.get_string()))) - { - return false; - } - } - - // parse separator (:) - if (JSON_UNLIKELY(get_token() != token_type::name_separator)) - { - return sax->parse_error(m_lexer.get_position(), - m_lexer.get_token_string(), - parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::name_separator, "object separator"))); - } - - // parse values - get_token(); - continue; + return false; } - // closing } - if (JSON_LIKELY(last_token == token_type::end_object)) - { - if (JSON_UNLIKELY(not sax->end_object())) - { - return false; - } - - // We are done with this object. Before we can parse a - // new value, we need to evaluate the new state first. - // By setting skip_to_state_evaluation to false, we - // are effectively jumping to the beginning of this if. - assert(not states.empty()); - states.pop_back(); - skip_to_state_evaluation = true; - continue; - } - else + // parse separator (:) + if (JSON_UNLIKELY(get_token() != token_type::name_separator)) { return sax->parse_error(m_lexer.get_position(), m_lexer.get_token_string(), parse_error::create(101, m_lexer.get_position(), - exception_message(token_type::end_object, "object"))); + exception_message(token_type::name_separator, "object separator"))); } + + // parse values + get_token(); + continue; } + + // closing } + if (JSON_LIKELY(last_token == token_type::end_object)) + { + if (JSON_UNLIKELY(not sax->end_object())) + { + return false; + } + + // We are done with this object. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + assert(not states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_object, "object"))); } } } @@ -17420,7 +17411,8 @@ class basic_json // add element to array (move semantics) m_value.array->push_back(std::move(val)); - // invalidate object + // invalidate object: mark it null so we do not call the destructor + // cppcheck-suppress accessMoved val.m_type = value_t::null; } From 710f26f95c82d8a14a18d717671fe83688f681aa Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 19 Mar 2019 10:06:35 +0100 Subject: [PATCH 047/220] :memo: added documentation --- doc/examples/json_pointer__operator_add.cpp | 23 ++++ doc/examples/json_pointer__operator_add.link | 1 + .../json_pointer__operator_add.output | 4 + .../json_pointer__operator_add_binary.cpp | 19 +++ .../json_pointer__operator_add_binary.link | 1 + .../json_pointer__operator_add_binary.output | 3 + include/nlohmann/detail/json_pointer.hpp | 118 ++++++++++++++++-- single_include/nlohmann/json.hpp | 118 ++++++++++++++++-- 8 files changed, 269 insertions(+), 18 deletions(-) create mode 100644 doc/examples/json_pointer__operator_add.cpp create mode 100644 doc/examples/json_pointer__operator_add.link create mode 100644 doc/examples/json_pointer__operator_add.output create mode 100644 doc/examples/json_pointer__operator_add_binary.cpp create mode 100644 doc/examples/json_pointer__operator_add_binary.link create mode 100644 doc/examples/json_pointer__operator_add_binary.output diff --git a/doc/examples/json_pointer__operator_add.cpp b/doc/examples/json_pointer__operator_add.cpp new file mode 100644 index 00000000..7520feb2 --- /dev/null +++ b/doc/examples/json_pointer__operator_add.cpp @@ -0,0 +1,23 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // create a JSON pointer + json::json_pointer ptr("/foo"); + std::cout << ptr << '\n'; + + // apppend a JSON Pointer + ptr /= json::json_pointer("/bar/baz"); + std::cout << ptr << '\n'; + + // append a string + ptr /= "fob"; + std::cout << ptr << '\n'; + + // append an array index + ptr /= 42; + std::cout << ptr << std::endl; +} diff --git a/doc/examples/json_pointer__operator_add.link b/doc/examples/json_pointer__operator_add.link new file mode 100644 index 00000000..9fc51060 --- /dev/null +++ b/doc/examples/json_pointer__operator_add.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__operator_add.output b/doc/examples/json_pointer__operator_add.output new file mode 100644 index 00000000..ae13afe2 --- /dev/null +++ b/doc/examples/json_pointer__operator_add.output @@ -0,0 +1,4 @@ +"/foo" +"/foo/bar/baz" +"/foo/bar/baz/fob" +"/foo/bar/baz/fob/42" diff --git a/doc/examples/json_pointer__operator_add_binary.cpp b/doc/examples/json_pointer__operator_add_binary.cpp new file mode 100644 index 00000000..620763e1 --- /dev/null +++ b/doc/examples/json_pointer__operator_add_binary.cpp @@ -0,0 +1,19 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // create a JSON pointer + json::json_pointer ptr("/foo"); + + // apppend a JSON Pointer + std::cout << ptr / json::json_pointer("/bar/baz") << '\n'; + + // append a string + std::cout << ptr / "fob" << '\n'; + + // append an array index + std::cout << ptr / 42 << std::endl; +} diff --git a/doc/examples/json_pointer__operator_add_binary.link b/doc/examples/json_pointer__operator_add_binary.link new file mode 100644 index 00000000..dabccf5c --- /dev/null +++ b/doc/examples/json_pointer__operator_add_binary.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__operator_add_binary.output b/doc/examples/json_pointer__operator_add_binary.output new file mode 100644 index 00000000..7536042c --- /dev/null +++ b/doc/examples/json_pointer__operator_add_binary.output @@ -0,0 +1,3 @@ +"/foo/bar/baz" +"/foo/fob" +"/foo/42" diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index 154e8b71..d4d6000a 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -56,8 +56,7 @@ class json_pointer @return a string representation of the JSON pointer - @liveexample{The example shows the result of `to_string`., - json_pointer__to_string} + @liveexample{The example shows the result of `to_string`.,json_pointer__to_string} @since version 2.0.0 */ @@ -79,6 +78,19 @@ class json_pointer /*! @brief append another JSON pointer at the end of this JSON pointer + + @param[in] ptr JSON pointer to append + @return JSON pointer with @a ptr appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::string) to append a reference token + @sa @ref operator/=(std::size_t) to append an array index + @sa @ref operator/(const json_pointer&, const json_pointer&) for a binary operator + + @since version 3.6.0 */ json_pointer& operator/=(const json_pointer& ptr) { @@ -88,14 +100,44 @@ class json_pointer return *this; } - /// @copydoc push_back(std::string&&) + /*! + @brief append an unescaped reference token at the end of this JSON pointer + + @param[in] token reference token to append + @return JSON pointer with @a token appended without escaping @a token + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + @sa @ref operator/=(std::size_t) to append an array index + @sa @ref operator/(const json_pointer&, std::size_t) for a binary operator + + @since version 3.6.0 + */ json_pointer& operator/=(std::string token) { push_back(std::move(token)); return *this; } - /// @copydoc operator/=(std::string) + /*! + @brief append an array index at the end of this JSON pointer + + @param[in] array_index array index ot append + @return JSON pointer with @a array_index appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + @sa @ref operator/=(std::string) to append a reference token + @sa @ref operator/(const json_pointer&, std::string) for a binary operator + + @since version 3.6.0 + */ json_pointer& operator/=(std::size_t array_index) { return *this /= std::to_string(array_index); @@ -103,15 +145,39 @@ class json_pointer /*! @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + + @param[in] lhs JSON pointer + @param[in] rhs JSON pointer + @return a new JSON pointer with @a rhs appended to @a lhs + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a lhs and @a rhs. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + + @since version 3.6.0 */ - friend json_pointer operator/(const json_pointer& left_ptr, - const json_pointer& right_ptr) + friend json_pointer operator/(const json_pointer& lhs, + const json_pointer& rhs) { - return json_pointer(left_ptr) /= right_ptr; + return json_pointer(lhs) /= rhs; } /*! @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] token reference token + @return a new JSON pointer with unescaped @a token appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::string) to append a reference token + + @since version 3.6.0 */ friend json_pointer operator/(const json_pointer& ptr, std::string token) { @@ -120,10 +186,22 @@ class json_pointer /*! @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] array_index array index + @return a new JSON pointer with @a array_index appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::size_t) to append an array index + + @since version 3.6.0 */ - friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) + friend json_pointer operator/(const json_pointer& ptr, std::size_t array_index) { - return json_pointer(lhs) /= array_index; + return json_pointer(ptr) /= array_index; } /*! @@ -790,12 +868,34 @@ class json_pointer return result; } + /*! + @brief compares two JSON pointers for equality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is equal to @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { return lhs.reference_tokens == rhs.reference_tokens; } + /*! + @brief compares two JSON pointers for inequality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is not equal @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ friend bool operator!=(json_pointer const& lhs, json_pointer const& rhs) noexcept { diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 7509df49..c96ddc4d 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -8466,8 +8466,7 @@ class json_pointer @return a string representation of the JSON pointer - @liveexample{The example shows the result of `to_string`., - json_pointer__to_string} + @liveexample{The example shows the result of `to_string`.,json_pointer__to_string} @since version 2.0.0 */ @@ -8489,6 +8488,19 @@ class json_pointer /*! @brief append another JSON pointer at the end of this JSON pointer + + @param[in] ptr JSON pointer to append + @return JSON pointer with @a ptr appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::string) to append a reference token + @sa @ref operator/=(std::size_t) to append an array index + @sa @ref operator/(const json_pointer&, const json_pointer&) for a binary operator + + @since version 3.6.0 */ json_pointer& operator/=(const json_pointer& ptr) { @@ -8498,14 +8510,44 @@ class json_pointer return *this; } - /// @copydoc push_back(std::string&&) + /*! + @brief append an unescaped reference token at the end of this JSON pointer + + @param[in] token reference token to append + @return JSON pointer with @a token appended without escaping @a token + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + @sa @ref operator/=(std::size_t) to append an array index + @sa @ref operator/(const json_pointer&, std::size_t) for a binary operator + + @since version 3.6.0 + */ json_pointer& operator/=(std::string token) { push_back(std::move(token)); return *this; } - /// @copydoc operator/=(std::string) + /*! + @brief append an array index at the end of this JSON pointer + + @param[in] array_index array index ot append + @return JSON pointer with @a array_index appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + @sa @ref operator/=(std::string) to append a reference token + @sa @ref operator/(const json_pointer&, std::string) for a binary operator + + @since version 3.6.0 + */ json_pointer& operator/=(std::size_t array_index) { return *this /= std::to_string(array_index); @@ -8513,15 +8555,39 @@ class json_pointer /*! @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + + @param[in] lhs JSON pointer + @param[in] rhs JSON pointer + @return a new JSON pointer with @a rhs appended to @a lhs + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a lhs and @a rhs. + + @sa @ref operator/=(const json_pointer&) to append a JSON pointer + + @since version 3.6.0 */ - friend json_pointer operator/(const json_pointer& left_ptr, - const json_pointer& right_ptr) + friend json_pointer operator/(const json_pointer& lhs, + const json_pointer& rhs) { - return json_pointer(left_ptr) /= right_ptr; + return json_pointer(lhs) /= rhs; } /*! @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] token reference token + @return a new JSON pointer with unescaped @a token appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::string) to append a reference token + + @since version 3.6.0 */ friend json_pointer operator/(const json_pointer& ptr, std::string token) { @@ -8530,10 +8596,22 @@ class json_pointer /*! @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] array_index array index + @return a new JSON pointer with @a array_index appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa @ref operator/=(std::size_t) to append an array index + + @since version 3.6.0 */ - friend json_pointer operator/(const json_pointer& lhs, std::size_t array_index) + friend json_pointer operator/(const json_pointer& ptr, std::size_t array_index) { - return json_pointer(lhs) /= array_index; + return json_pointer(ptr) /= array_index; } /*! @@ -9200,12 +9278,34 @@ class json_pointer return result; } + /*! + @brief compares two JSON pointers for equality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is equal to @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ friend bool operator==(json_pointer const& lhs, json_pointer const& rhs) noexcept { return lhs.reference_tokens == rhs.reference_tokens; } + /*! + @brief compares two JSON pointers for inequality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is not equal @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ friend bool operator!=(json_pointer const& lhs, json_pointer const& rhs) noexcept { From e07e8e791250328e97c21c79f5a341664d835e3f Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 19 Mar 2019 15:40:23 +0100 Subject: [PATCH 048/220] :memo: updated documentation --- doc/examples/json_pointer__back.cpp | 15 +++++++ doc/examples/json_pointer__back.link | 1 + doc/examples/json_pointer__back.output | 2 + doc/examples/json_pointer__pop_back.cpp | 21 ++++++++++ doc/examples/json_pointer__pop_back.link | 1 + doc/examples/json_pointer__pop_back.output | 4 ++ include/nlohmann/detail/json_pointer.hpp | 43 ++++++++++++++++--- include/nlohmann/json.hpp | 6 ++- single_include/nlohmann/json.hpp | 49 ++++++++++++++++++---- 9 files changed, 126 insertions(+), 16 deletions(-) create mode 100644 doc/examples/json_pointer__back.cpp create mode 100644 doc/examples/json_pointer__back.link create mode 100644 doc/examples/json_pointer__back.output create mode 100644 doc/examples/json_pointer__pop_back.cpp create mode 100644 doc/examples/json_pointer__pop_back.link create mode 100644 doc/examples/json_pointer__pop_back.output diff --git a/doc/examples/json_pointer__back.cpp b/doc/examples/json_pointer__back.cpp new file mode 100644 index 00000000..3d57c589 --- /dev/null +++ b/doc/examples/json_pointer__back.cpp @@ -0,0 +1,15 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // different JSON Pointers + json::json_pointer ptr1("/foo"); + json::json_pointer ptr2("/foo/0"); + + // call empty() + std::cout << "last reference token of " << ptr1 << " is " << ptr1.back() << '\n' + << "last reference token of " << ptr2 << " is " << ptr2.back() << std::endl; +} diff --git a/doc/examples/json_pointer__back.link b/doc/examples/json_pointer__back.link new file mode 100644 index 00000000..adca23d7 --- /dev/null +++ b/doc/examples/json_pointer__back.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__back.output b/doc/examples/json_pointer__back.output new file mode 100644 index 00000000..da4d0279 --- /dev/null +++ b/doc/examples/json_pointer__back.output @@ -0,0 +1,2 @@ +last reference token of "/foo" is foo +last reference token of "/foo/0" is 0 diff --git a/doc/examples/json_pointer__pop_back.cpp b/doc/examples/json_pointer__pop_back.cpp new file mode 100644 index 00000000..ed3417ec --- /dev/null +++ b/doc/examples/json_pointer__pop_back.cpp @@ -0,0 +1,21 @@ +#include +#include + +using json = nlohmann::json; + +int main() +{ + // create empty JSON Pointer + json::json_pointer ptr("/foo/bar/baz"); + std::cout << ptr << '\n'; + + // call pop_back() + ptr.pop_back(); + std::cout << ptr << '\n'; + + ptr.pop_back(); + std::cout << ptr << '\n'; + + ptr.pop_back(); + std::cout << ptr << '\n'; +} diff --git a/doc/examples/json_pointer__pop_back.link b/doc/examples/json_pointer__pop_back.link new file mode 100644 index 00000000..a9f9c88b --- /dev/null +++ b/doc/examples/json_pointer__pop_back.link @@ -0,0 +1 @@ +online \ No newline at end of file diff --git a/doc/examples/json_pointer__pop_back.output b/doc/examples/json_pointer__pop_back.output new file mode 100644 index 00000000..b0468dc5 --- /dev/null +++ b/doc/examples/json_pointer__pop_back.output @@ -0,0 +1,4 @@ +"/foo/bar/baz" +"/foo/bar" +"/foo" +"" diff --git a/include/nlohmann/detail/json_pointer.hpp b/include/nlohmann/detail/json_pointer.hpp index d4d6000a..465e5165 100644 --- a/include/nlohmann/detail/json_pointer.hpp +++ b/include/nlohmann/detail/json_pointer.hpp @@ -210,7 +210,7 @@ class json_pointer @return parent of this JSON pointer; in case this JSON pointer is the root, the root itself is returned - @complexity Constant. + @complexity Linear in the length of the JSON pointer. @liveexample{The example shows the result of `parent_pointer` for different JSON Pointers.,json_pointer__parent_pointer} @@ -230,19 +230,50 @@ class json_pointer } /*! - @brief remove and return last reference token + @brief remove last reference token + + @pre not `empty()` + + @liveexample{The example shows the usage of `pop_back`.,json_pointer__pop_back} + + @complexity Constant. + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 */ - std::string pop_back() + void pop_back() { if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } - auto last = reference_tokens.back(); reference_tokens.pop_back(); - return last; + } + + /*! + @brief return last reference token + + @pre not `empty()` + @return last reference token + + @liveexample{The example shows the usage of `back`.,json_pointer__back} + + @complexity Constant. + + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 + */ + const std::string& back() + { + if (JSON_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); + } + + return reference_tokens.back(); } /*! @@ -255,7 +286,7 @@ class json_pointer @liveexample{The example shows the result of `push_back` for different JSON Pointers.,json_pointer__push_back} - @since version 0.6.0 + @since version 3.6.0 */ void push_back(const std::string& token) { diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index 6a5c1075..fd4b09e7 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -7507,7 +7507,8 @@ class basic_json } // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); + const auto last_path = ptr.back(); + ptr.pop_back(); basic_json& parent = result[ptr]; switch (parent.m_type) @@ -7552,7 +7553,8 @@ class basic_json const auto operation_remove = [&result](json_pointer & ptr) { // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); + const auto last_path = ptr.back(); + ptr.pop_back(); basic_json& parent = result.at(ptr); // remove child diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index c96ddc4d..0f18cfb0 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -8620,7 +8620,7 @@ class json_pointer @return parent of this JSON pointer; in case this JSON pointer is the root, the root itself is returned - @complexity Constant. + @complexity Linear in the length of the JSON pointer. @liveexample{The example shows the result of `parent_pointer` for different JSON Pointers.,json_pointer__parent_pointer} @@ -8640,19 +8640,50 @@ class json_pointer } /*! - @brief remove and return last reference token + @brief remove last reference token + + @pre not `empty()` + + @liveexample{The example shows the usage of `pop_back`.,json_pointer__pop_back} + + @complexity Constant. + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 */ - std::string pop_back() + void pop_back() { if (JSON_UNLIKELY(empty())) { JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); } - auto last = reference_tokens.back(); reference_tokens.pop_back(); - return last; + } + + /*! + @brief return last reference token + + @pre not `empty()` + @return last reference token + + @liveexample{The example shows the usage of `back`.,json_pointer__back} + + @complexity Constant. + + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 + */ + const std::string& back() + { + if (JSON_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent")); + } + + return reference_tokens.back(); } /*! @@ -8665,7 +8696,7 @@ class json_pointer @liveexample{The example shows the result of `push_back` for different JSON Pointers.,json_pointer__push_back} - @since version 0.6.0 + @since version 3.6.0 */ void push_back(const std::string& token) { @@ -20241,7 +20272,8 @@ class basic_json } // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); + const auto last_path = ptr.back(); + ptr.pop_back(); basic_json& parent = result[ptr]; switch (parent.m_type) @@ -20286,7 +20318,8 @@ class basic_json const auto operation_remove = [&result](json_pointer & ptr) { // get reference to parent of JSON pointer ptr - const auto last_path = ptr.pop_back(); + const auto last_path = ptr.back(); + ptr.pop_back(); basic_json& parent = result.at(ptr); // remove child From 18cc7ddd62d7c2cd3a5f74a4db75c43d122966d1 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 19 Mar 2019 16:03:02 +0100 Subject: [PATCH 049/220] :memo: completed documentation index page --- doc/index.md | 24 ++++++++++++++++++++++-- include/nlohmann/json.hpp | 2 +- single_include/nlohmann/json.hpp | 2 +- 3 files changed, 24 insertions(+), 4 deletions(-) diff --git a/doc/index.md b/doc/index.md index ea6dc6ae..42481c3d 100644 --- a/doc/index.md +++ b/doc/index.md @@ -35,6 +35,12 @@ These pages contain the API documentation of JSON for Modern C++, a C++11 header - @link nlohmann::basic_json::out_of_range out_of_range @endlink for exceptions indicating access out of the defined range - @link nlohmann::basic_json::other_error other_error @endlink for exceptions indicating other library errors - lexicographical comparison operators + - @link nlohmann::basic_json::operator== operator== @endlink + - @link nlohmann::basic_json::operator!= operator!= @endlink + - @link nlohmann::basic_json::operator< operator<= @endlink + - @link nlohmann::basic_json::operator<= operator< @endlink + - @link nlohmann::basic_json::operator> operator> @endlink + - @link nlohmann::basic_json::operator>= operator>= @endlink - serialization - @link nlohmann::basic_json::dump dump @endlink serialize to string - @link nlohmann::basic_json::operator<<(std::ostream&, const basic_json &) operator<< @endlink serialize to stream @@ -44,10 +50,12 @@ These pages contain the API documentation of JSON for Modern C++, a C++11 header - @link nlohmann::basic_json::operator>>(std::istream&, basic_json&) operator>> @endlink parse from stream - @link nlohmann::basic_json::accept accept @endlink check for syntax errors without parsing - @link nlohmann::json_sax SAX interface @endlink define a user-defined SAX event consumer + - @link nlohmann::basic_json::parser_callback_t callback interface @endlink register a callback to the parse function - [binary formats](binary_formats.md): - CBOR: @link nlohmann::basic_json::from_cbor from_cbor @endlink / @link nlohmann::basic_json::to_cbor to_cbor @endlink - MessagePack: @link nlohmann::basic_json::from_msgpack from_msgpack @endlink / @link nlohmann::basic_json::to_msgpack to_msgpack @endlink - UBJSON: @link nlohmann::basic_json::from_ubjson from_ubjson @endlink / @link nlohmann::basic_json::to_ubjson to_ubjson @endlink + - BSON: @link nlohmann::basic_json::from_bson from_bson @endlink / @link nlohmann::basic_json::to_bson to_bson @endlink - Types - @link nlohmann::basic_json::array_t arrays @endlink - @link nlohmann::basic_json::object_t objects @endlink @@ -57,7 +65,10 @@ These pages contain the API documentation of JSON for Modern C++, a C++11 header - @link nlohmann::basic_json::number_integer_t signed integers @endlink - @link nlohmann::basic_json::number_unsigned_t unsigned integers @endlink - @link nlohmann::basic_json::number_float_t floating-point @endlink -- @link nlohmann::json_pointer JSON Pointer @endlink +- further JSON standards + - @link nlohmann::json_pointer JSON Pointer @endlink (REF 6901) + - @link nlohmann::basic_json::patch JSON Patch @endlink (RFC 6902) + - @link nlohmann::basic_json::merge_patch JSON Merge Patch @endlink (RFC 7396) # Container function overview @@ -282,7 +293,7 @@ Note that this table only lists those exceptions thrown due to the type. For ins @link nlohmann::basic_json::swap `swap` @endlink - lookup + lookup `find` @link nlohmann::basic_json::find `find` @endlink @link nlohmann::basic_json::find `find` @endlink (returns `end()`) @@ -300,6 +311,15 @@ Note that this table only lists those exceptions thrown due to the type. For ins @link nlohmann::basic_json::count `count` @endlink (returns `0`) @link nlohmann::basic_json::count `count` @endlink (returns `0`) + + `contains` + @link nlohmann::basic_json::contains `contains` @endlink + @link nlohmann::basic_json::contains `contains` @endlink (returns `false`) + @link nlohmann::basic_json::contains `contains` @endlink (returns `false`) + @link nlohmann::basic_json::contains `contains` @endlink (returns `false`) + @link nlohmann::basic_json::contains `contains` @endlink (returns `false`) + @link nlohmann::basic_json::contains `contains` @endlink (returns `false`) + @copyright Copyright © 2013-2018 Niels Lohmann. The code is licensed under the [MIT License](http://opensource.org/licenses/MIT). diff --git a/include/nlohmann/json.hpp b/include/nlohmann/json.hpp index fd4b09e7..f6ce9bb6 100644 --- a/include/nlohmann/json.hpp +++ b/include/nlohmann/json.hpp @@ -3990,7 +3990,7 @@ class basic_json template bool contains(KeyT&& key) const { - return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); + return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); } /// @} diff --git a/single_include/nlohmann/json.hpp b/single_include/nlohmann/json.hpp index 0f18cfb0..41070010 100644 --- a/single_include/nlohmann/json.hpp +++ b/single_include/nlohmann/json.hpp @@ -16755,7 +16755,7 @@ class basic_json template bool contains(KeyT&& key) const { - return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); + return is_object() and m_value.object->find(std::forward(key)) != m_value.object->end(); } /// @} From a6f9b4e36da52b906d3dbef1b31d859164be9672 Mon Sep 17 00:00:00 2001 From: Niels Lohmann Date: Tue, 19 Mar 2019 17:18:33 +0100 Subject: [PATCH 050/220] :busts_in_silhouette: added contributors --- README.md | 23 +++++++++++++++++++++++ doc/avatars.png | Bin 831937 -> 892059 bytes 2 files changed, 23 insertions(+) diff --git a/README.md b/README.md index 8acb2842..52744e6f 100644 --- a/README.md +++ b/README.md @@ -1231,6 +1231,29 @@ I deeply appreciate the help of the following people. - [Sylvain Corlay](https://github.com/SylvainCorlay) added code to avoid an issue with MSVC. - [mefyl](https://github.com/mefyl) fixed a bug when JSON was parsed from an input stream. - [Millian Poquet](https://github.com/mpoquet) allowed to install the library via Meson. +- [Michael Behrns-Miller](https://github.com/moodboom) found an issue with a missing namespace. +- [Nasztanovics Ferenc](https://github.com/naszta) fixed a compilation issue with libc 2.12. +- [Andreas Schwab](https://github.com/andreas-schwab) fixed the endian conversion. +- [Mark-Dunning](https://github.com/Mark-Dunning) fixed a warning in MSVC. +- [Gareth Sylvester-Bradley](https://github.com/garethsb-sony) added `operator/` for JSON Pointers. +- [John-Mark](https://github.com/johnmarkwayve) noted a missing header. +- [Vitaly Zaitsev](https://github.com/xvitaly) fixed compilation with GCC 9.0. +- [Laurent Stacul](https://github.com/stac47) fixed compilation with GCC 9.0. +- [Ivor Wanders](https://github.com/iwanders) helped reducing the CMake requirement to version 3.1. +- [njlr](https://github.com/njlr) updated the Buckaroo instructions. +- [Lion](https://github.com/lieff) fixed a compilation issue with GCC 7 on CentOS. +- [Isaac Nickaein](https://github.com/nickaein) improved the integer serilization performance and implemented the `contains()` function. +- [past-due](https://github.com/past-due) suppressed an unfixable warning. +- [Elvis Oric](https://github.com/elvisoric) improved Meson support. +- [Matěj Plch](https://github.com/Afforix) fixed an example in the README. +- [Mark Beckwith](https://github.com/wythe) fixed a typo. +- [scinart](https://github.com/scinart) fixed bug in the serializer. +- [Patrick Boettcher](https://github.com/pboettch) implemented `push_back()` and `pop_back()` for JSON Pointers. +- [Bruno Oliveira](https://github.com/nicoddemus) added support for Conda. +- [Michele Caini](https://github.com/skypjack) fixed links in the README. +- [Hani](https://github.com/hnkb) documented how to install the library with NuGet. +- [Mark Beckwith](https://github.com/wythe) fixed a typo. +- [yann-morin-1998](https://github.com/yann-morin-1998) helped reducing the CMake requirement to version 3.1. Thanks a lot for helping out! Please [let me know](mailto:mail@nlohmann.me) if I forgot someone. diff --git a/doc/avatars.png b/doc/avatars.png index 49baa70a05e934b28a98d5dd241ae29f7723b582..e729b01dee215daec268477cafd24fea85a6c6ac 100644 GIT binary patch delta 758981 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 z%0D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%0DWMn8@B^kjXB~j*f*_OjHwbaGLOwN&G#MDTm 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%0A)gl5GX8WsE7UWpoS%PDl>#4NI2ZPRq6KYD9Jl6 z0eSPeZr>w!{rqLCZd%>C_0ZU|>b4!<{+F#=w%+oI&rTnn|Key+Q>a|+4l61RS z5^MIh7Mo?w7KSbgYOO&Wkx~kk(nJIhgn``9sK_uGh@>bB2II!+Pov=7FV4 zq=xYr@^CLB1pr6{3PCAW3bmF9qeSi4?QYmnz5m1S-@R+!>)-d^PdnS*@zt+?eC|`9 z$+M5!$HOjN@VoZz1%SPK_X5D!HQxoncm2|DUUBx|JIxL6y67Mw?(5i-PF^-O!{2#u zFAxF4_X(biFzXM}dR-|6!7C#63?QVmw>-XV% zD2i01q!O!%X2pq5n-hc-rRgJoyLL1CWk~`cq(~?pbmr@8PCDa9etg?seSjktwpxG~ zY>{VqZh(S!3+;u%`&3!Ie3>??ti20j2#$mILM(#BtdtNDVM!RR%S}oE7CG&P&P`ba z6+lFds1QMr${QC~Z`1m~r=RI|021A31Q_J&%rmeEJhp(+6+9 z?XKZXqhVYJN5DOcEjs`*JKqJ;|L|wO|LQl~{&PR|>;Lqhe|7hvgYW*ax7`{2%8uig7jaj9?Q4e)&kejkYs2z7WEC3e zdC!8Qsadytq!Q7i5ASS$q>UmOqz0Rn*fd%ua;QiUiO3m~&mBDJ?31=!@Y3Gg?)iR^ z5xKm`7jlm#N~_IEl_{O=W-CTD7(*rkMJkFwsmAiP)FhA~%);91)PW>PoaK3vWg0;p z&30M_kh19gAZk&ne2tWgUIv*|W@exayMQ#H0;R~9ptUn;b6J&tBlymuK0Dy=LA{;2Lj;}o%Rkh!pY0r1_%(7C`y`t5MT4u1U)d-*~ zic>|)Mn*3_@9dR-W0jlkxM#L!UwO%yhZbfRI`f*KBr7x~&Uyog4qp7~3%~od85^M8 z?h*@;F`%6drV-_NAw+^`;?!fMt!h4B%y0DMwHwM5r|wHMYKg`O=q-oPEZ~_SxDQgKNI{ z`Au#8i6atb$6~xK+1xu@f*t#O+ zdaNT~VnvVywe$JJ$nqqugV?|>x0VQFBRgj1S2k8oG+PA-D5a6C^Ue;&T8+#W(}xyn z)r0<(wX`08^@==htTu7XK>&z!cW(C3n)S;oqpR~n4^}Hlr;~BuW`(rYM4^&`js$ET zZ0>9ZF6X|m5#OD4y`J@q0Ny&mgo%Umh~(_x2=#|OAECI4 zQAO#PB7?@Di6}B;B4r{p5gJV<(oqCDYOEaj`PaCA=(fLKxBp*n{M;wLcIDRs3kV!* zbxtfL&`l}TUYMJk>Of^uc%Y17MNA4w1K~4(1O4A7h{B8fB!EMvMnn(}&GBVuQXg!i z(K`9m4O!MBQXJs+UDHratF@?fZxujqzEkAgdLuOvtytNlNM~*UioEx}a1uhE_w&KP zWmbHDf7$Bs`EEZ=>eXt)B+0JR6-gkBBe-~WX1S~*PIqFYf%zoCDn0-k`!t+U9GDmrZ}QltH`Lx zU~J-Qx@*7xySvuy>#7g_!S7x9&F_?;%x8Rmf0+X_eClH#sc-v>i_TwO)LSE4ChHT; zqf2-%Xh9vZVCD&%P@iXjqK!yauR02nfMBMK0b z0;Q0&5TQae5OTNicU%Tp-QT^f+|(3szdo*N(1OH@0kT zePm1}Nqua2eSEUg7;88#^Mj$|qN+Ct4 zjH*PnhB^I$Wh>8Iz3SXm+^W}Bk6!e$<(s#zFw4d&D@K!L4PC3~Y8u5+(x|`eOx?Zu zPe1%;e>li;M11zS;u1l=XZOy3kNnEp;P(IZ#qnS2a^@-)9t&xVX z&wK5nJDay>LS9IuyzL+Yk|_oYD<)U`$}j!wmXp`6Tf2Jg>gDWwtfP7*CJ@iI-R)0K zjt{ysXxo88Ht2)*09=-{l_o`~2#wNFebtK9>#LE8qSyzswW)!dGqFzzx?^H zK6KB``=)z;^Xa=o>#T{5=hsG-=UHy;AUJP~Rz!+Dpf53@APg*8hK)Q5frJ!lf<-JO z5P{acL!H}iJapbg6YqROD@x2~uj}0RNU@Nozkb=F4<^f2t*&2x{o#l1@6;+9B&yXh zVDiYJY>23=jzd z0Rn2&5gG%UfRG|OymN3Q|KU&k@f%8U5x2)CJbZL6s z=FQjLG;K7k*;KE8Pha)*uY47eo+e!I$)#FL_M5Hu-1T4rK2Uk`W8OmKINgQS&(24F;k`e- ze#54VFS+bBm;cDxt*`A&A2|5%t-Bt)JDZ!6pn_6}V4_%;2p%DVk{A^gDupHzqtR#; zX)!uzvtmVmr4r45{8L|FbNjBbk;X0Gy{TWQZ{K)Zq|Ie-eC=C*@CR$@+26U>u3u&z zxaGcnRscRNX!(;!iF3H`u6us$@@wAt?(;tR{wWZIC6YiwItJ29B^U+KfUvM-pDhh( z^|?sUQH}Z7k*EN|DoOx=eTJx(oVp4Eq*E1D1@+Q@Ix2zv<3am@1pq~i(L_`Np`%JA z9^E1sqi#^t1U3p0$beP=a{X0T-hS(S22{18M@DLKWGa>9^fOOcwr;J{sxKZ;1q7{G zM3pThmGRRn&Aa~(&*$GRbRdPaPN{UHa1!#2lnmz&bO{YB5N%I`rx#V z@4owgzFTg-?JHmU(m$Ve#>+2$=@0+tyNi0hE8myfd{r%0!FmA!(W11VD#2ew29d}Z zQCf6l(g=)MbL#lI4fW-#G&tOR%1PH;f5(x}K9~jo*!lM#zy7ML-}nFjFZLz>ep%eU z>Gr!qV4^3|`8+E@5SA#33MU75EV!(oC3Iweh*@^oEH0r#0;t$$QFTRBnOs6Y5QJGr zwS~Rc9_Jzi$bfKl>{KLkY-ovy!oD|sZ{FL##LQ&`VOCMCGJ49 ztK*~|RpLgG&65h+RO*u#NM!>8e(5uR`|^ulo>r^(Tyvd|mhYOK|LWJj{=WD9r!Rl? z>$9^5jnU3|B5=-?6W^moLqzVp`>yQ|ZU5NEJ~1~vr_*ZMY`yH%4KF#RX(b?l2GU3h z%03`cq9P{MT)Qk9NxO3yM!L`Oxm%lmCms00pZ?N^?t9>2QdBmP9YK>3sfYF+{=?t+ z&0qV+Q*|=>(T5&{BNdA0Ob}5fCntO3=Y8u_4=D(IEEZ_7YofF7y5nP(VdEvg8@D!y zd*Waq3{k-Y6JKmFZnF8EbnbnuDTXF=4#b|1O+H$u^WUJ`T7 zzDP&5YXwNW!sg2COq?cwE< zXRI7gZRRxT1QA&YC{4>gJo;Nvfq%A#cC)tsCxp;9;dGgrf$64oRH7 zQu^Th;F`bt%OClfpPlVw_}teyQ3RH4*fcoU(Ut=PG9UzyCw$R`bgWE&m6lSVvY$hQ zg}prSz!K~106>{49!G(cFdzy$k#LOrRhD7`=@i(*@j(0fiJ*P&ix4b8OiCH85QIq8 z$JQ@iFrXq|EZl$lw-*jS)T~raK4nWbuu^1-vs$A)=q$u>T&va?gDnadf>yv;&Y@Ri z#l#tx9_~aN*2ew8+($lt{MXG|b;rXG96UU~d;dWYwPJGpfZCt`?5B1g*pJ3Xk@LhM z!zhZJ8>V@fhhNkvipo_mMrB#pwRdjaTDZ@%supYvI9`qs0mt-2!L7*Rzp3pQ(xP6peHC@=t`sZ?4FXc@S0 zdAApIdi@>SH*G$D>4xjS`7fXN)SlgY-+1{erVi{|9*-VAG!>boHL@bc?&KtF-MV7F z-`#%y&ThZ2jdr%|fyX0k@gcA=s%#t$ff=}$c@ZzbT9O1+43QEaSgB~N61VD-z=l&- zpAzo;#Q*x;%ILC}TyWu`*`6G~>7Z;r?DcwI{LE0v@`1Ux>AZo_U}K? z99wtz(BX}LD<{PnMV)x@|uwAZa(xpw}x2a7yJ8kJTd1YlMOBEsI2 zD#smvD=0fJn6o1D4k#$S$h*KufTBVhN7+j)ci*t*7tZJ*Q)g(0fLu4?mo2 zyQ2n)nSI;E@B4#4sQ%7xrQ^$A_KHjX)35#)L@aICaqk7e`c)%4_Dsoh^1-J-{16;Gy1f*Jalu{DCN-}ZB$>-!oim1Led&hbW~CCH~=7`cg6hdOcFPom+^Yt z7{d~BcA(?5(gK7KM8NCDNWIat`M?Xu6R{eZ=pO2{`53!Ca)hd8I_{h!& zb}ta}Ah(g0FMRgXw_FRCU-q)*GL2Y&v95g-)(>EjA)c8LP!^+a3DV7q+>6nmvB<<0 zk}EVsOm@!kz;dm=^S)i%UcUCS|Nc>qCn|$SCr2uW=X-E`#X!VHwf?3zzxMrq`O!sH zdu{{?Md-TEeE3Iy^Sr|cMmyX05s^QtmG%WK!pwq!83jETQFDcg>Sf}zvoLFa2SpsE zNt~3m?wNxJop7_!YPGa;8KR2g7y*24i@c2}(+Brj$3Efl*T0~xU=_MDw z?nh{-zt7py3JakubO-zH{`%`Kt(tlYKtRVUSv-Z)9SPGO|0r1^5fvgG=dsA>aJ<8x ze)c_1n{nDmD%G@FPwJKW@=e2d4PgJiM}oDLY9)@7dbL4X0dTCTJv%jjbNE1!W#TQe zvo;4!I;lVuRU0Ffq!z`ocRn~K5n=YhRjbu%6c@Q&==QBIn4Lm&d1jr9E6L1!XUCqs z9PBW;H5{D808mbhl#(8-B-|jSlvW6+P=GZ%C*+YxF$rlDVJ$_w_uVV^^yl({p~i+~ zPLfM6yYQa7@4fTh`xU8wMWU)9n<6O{yk9k1eeF;G;-Pqa*+}iseDAq6LI8k()4Ly9 zad|T{m54G#gtGDWX$t97nI%dMBCMkX$#~l)2}>suiw*$lC=#hInMi~sd0|c{jO4w` zYzP74xIVe!WGIWrotfPa+$F96g67y*0Eo5j4?5kcwgwErRqGpnt+RQa#YvnbsWOQj zxOS(@!p&wK_aFirg_fCxYn9X}>=6J(GfN5PCVU z9MuJjIAdW(5}QlA2WwX?*B}mr07Rq&EHVlpqeuu6kmExS%e#7k&i_@BC5x$nqIl0ue?GtaY8};= zE~WyKPM($#(%O(Ag8;yl+6d|j3InQoW3Md(XspeX3vi*WKgS-#BHo-gl6G@QgEB)yhJD@En#i z5K)-VIPaWCE0=%vpC3r2WFCGdLZ-6x@**PYatk{SLm87WAnK^TWU*}cs;B7&;o{;E zaeaIxmW)0K)@Fkuh%&WiBV>gm0%Lmp4*T4H6$3!%^ybFxYAmz>q1G7bV8KbKkF)~2 zZeinUwFusU0AR1z=O73eLQu*eg0(JiAoUokvmAROj}7b|Iw~dCt}%wnHXOvVt)@LU z-CY=`p`#593=mKlfjI;Z-h*=(LI}8eL-m}q>R_rH-u*{kJ#gwD{pZUVL?8bQPfxUtl+p}i?7jEJ)fGu=)b zM0%asUZ>q#m(`5_ty%^nd~qBOpT%L}JCBTNCkF7me%aI%|7qXMzFQXPIjy)oU!s)K+EB1qskFvc$S{QnXb-WS*XuPK zbtld!wl+_+2_8HLn+X3_%zihs z$Yl|C!^X{T`;Gs-YX-i0w~b=8So(ayHx?pt2tdGTbyTHevREPx1=u2gi_*k97eKOh zJ4Q9us!q?f2}v7>h=a?r{(>>fAgCg9A%wzN=L;q*obUDeGF)18-h1!C20|2&5S()b zpkiP|WR~G50WMqpN=+JuV2ds8B!Xda?jRBg0wQSyDEV4I7=Q!f6VXHD>JK!y3RKhT(vuXQ)Gq9|m4~ z>ACmrjXwM3g^5NxKQNCJO32D{vy9K5AQCBs65`r2pgKZ{W`KOq>GitIN~uVgkf_$I zsrpDi&jWZ5p(y&==%TL*6A=lsWs!cbQ)GQx+RQEbLCav093~S zcb=*ckqEr~T|e1>JNuVzzUp9_wdW6YL%#>W5cq}P*$k)vP$7r3wh|WS5ecxV#^Vzc z`}Q1c)SLC@C`GE#8mp(3Zl@b4FK4k8g8={*0~LHBXqb6+=AgHkD{NII zt6y`T5s8lldY<6UA|hf4JT^K~uT?QWthD07q7Nj#^y9t{YGI&607*2Z696!bs1aT6 z=I0kA_#*FrnP^2?ZzPS0N)yH=rV5@L^fMyLx$wRyiU9-}6vf>9bkS=U{~ve%9c^iL z)rX>UuC>0fb2|IvQ#n_6RcCcjE1=XZBw;`p3G>E%vhg#%*oJqXjTz%&KkmI;+dL;8 z7h@YR$$+o{A&?px#cs7ask^FE<#_7ke0KiAT5HaK_s80MpHr=tKp=aVx1>?M&#qnj zeBZa`n(;S(vlnkSdW%bImW)fJFhXq7SQ`Ndtcj&gV?d_<>=7}*O7;Kj&vPL+umpub zYfKXNJl`V*1XLj6L|`;f3R;s+StqPh)Wq75Hlzu(Ws{Og z$ZE8IniDOw*<5J+$rD>Y`-@9YKehH+@aX$L{GBrsW6VHgg^+)q>XmC4bXxQsZ+~0V z)4%ldKR31hg$ExVPYfd(CIB)3_N{RBf5nz#KJl|fKQ{v4+zkf6t+f@l;CVi$X={B& zAo}^L$kwx!S}vPKAyciTQc|d>+itbCy6tU$GO?#1lP_dzBPa!v8$Jn$sFP%2VL=Mv zOp|qy4Le3}7!c3)2^rm!cOEn|TViW%6m{aLXSHD_DNzWawPqrLNX&L{l(CPdbBdHL zTa8A8jRmE`8bO9RVb)*?$PlGKmdG$AWK$r^sRgZ7(n;30O&q^t?ACwzxqowgZs8+; zA9?(dhwdhFzX*kJ9SA!~2q)=oVhl4gic&88{=f0D@Bhi4J8@*+=t#NMZcf$xgZGYV z3$BA4_$|kO4&Hx@|9_jug}(LTZ|wpEDQ3ppi1Z4(YlyjGMfq6*;e|Os5qQ$~Ln$lS zQr+`C8K_LYWY9wyWOF0=QoT|e8ylN{sZHz+3$?VD0?KMJKNvKCp71VRzaGbt6l^pb zZGnYD@({42sjxhlq2qWxiM1tdZK|~~7D<@IL{T@5BWn}o32T@c6|?iJC3AlQhD-CK z?avfp>%u*Y;0@@ekioi1JZPDTK?oKEXaEfAfD=#UIVhcq<;9jtNT9i*2M&rq`Ql+{s*gj_!grMz8#Bs^7IdRg-I~_>oSqZ zKtcp>Y8gX;LSR^!;-`hik@KI-OS|2%k{-}uO0Pdm@5!d zu25zItqnnJqIT5l_Ii;dHBu=oZ#1o=YGCq|5>f&%Su-RoaOZhYbRT37iHOMed=zM{ zwbr=p@qC|vyS*MtEEKDV0s&NiItXD$c~MHg#Z@7 zmVp5XK@wRsI*Igw>FgtW)4f0Rzciupwd-_dZsdP{--pV9$?re@@}GWYd$Ya1)l7A~ zys{}3j*m?R0ZxpLj*S*hedvGMzJ6}c;i>=ciAaFSlUnK&0QZbMD^3{z)m5A^n&8(o9aGN za_D)jIBB)JW0jGhIJ&j9m4{Ad@<6dPmJb`9ZkMHpLUoc}=m(&n*Y4>kA!}5am0={i z5(R)^Mj+yPsTu_R@v}jgb?rn>iY(co3;~0+6lStQC=>$N8lw?^1rOd2f^4h3j-aa9 zvQnUi&zHd>xUmNVoDExGaDKHz%rTRj*8+kWB`|{Z0i6gDyOC0045y_rFE;iwugZz3 zJO1+%o7Z1k3p2^`9#whxe|l}Bc;x#Zu(^@@s$0MJvp==4^c4@}V;^|;9e@4rpS_6% zuk*weI(7KOx@xR{otu-L?n|HF0*g~eDy!p@?VIaqrz^mMG4NKC4w&>eFjGuu*`|LH zq4-u7Q%WY`w-TU%fx7K|VXg9uLAa9va*K!z=}gA!CcULw*TPIj()p7w zdgZ)sZEz5H()WWf)9u6=-}ADWAd~U@K$PQN7o@O%h>U{z1EWBbWyVsW5U0kPIMEVC z$P7ZDQh}rGwM3o}08EyH&_^kyluAheY0}i#)RM6TmPL|Ap7MR=74yOqz8b=MhMi5W z9t4hA6P=?zpuk2)Qr7PkLJ;n+Py$$sghC**2F1O#MrXb4W&A5&?SK%WhZHBBPR|Q{ z8D=Jbro;P-H-7Qu=cSZPu8mdPI`hoq|Bny9_H>uKUDJ(Bq6HCa!`51+UexW}(f}IC z0Bd0Dl;vObWyu7Eq6mV4|Ij}H<#H{b3wn*nm}GKhtk>Dnjjdi|14#=Zw6;2qdKL^3 zAbZLYsWJmbX)jD;pVFLAd#hEb32jZ_2Zd07N=ucD-)yyeHp*l(Vo>8FqEu^RG@u1Q zOf3i@g|NiF^n)M=DH2-%m&;*o60q{{WYus_0RTH&umhbZWN-q*AQ;5Jrcu8S!jZND z_02ZA^r>~IEEdQ%MYK}+lvpVRSgEZ9DZp2Zn1J9Y%8%@+)~D>%B`E|kF>|qy-;M)+ zqZ#7FrkZU^sYb)ZmdOGWf(C7X0NEn{MG=YrgDqopwlD(1gF$r$5Er2blFare2%PP@oZ2QWq z6|!56t<8-{qA>jiS@b+lN@Z=@zXY~_MjHzVHf3d{@>0J*VaO#SHy0)${8fs_QWj4I)`pg0(Jx!-)h8 zG@QZvS9+zEtJzVw9oU3`fU-7qqBe1`(^^K3R+n#d8ygvs%%xksC^Z&1CBZ;Mh7lMn zF%ht(dVQ=*RvYbS{9ZF+OJOz_`d*r*iH)OfQ$g$t)a05jthI(ndrAqA{rr$KEkxEb znXVAA3*Q`|{^*|a+fE&rsONQmnt%)xfD=Qx(+xP7B|1O= z67*4enulW&EHNq|MnQu?5KIJs^uXQW)J$%>@)Q5)?f>^L{KtR(+6&MA@W1@k-+ZNb zfFh0~lOzHRScAqe6EcG}K$h7eF(MH#!XVf3SA3;b1)1_&{~a@kAYa~p2LSuahsebC zRJ;D_pSg$3FJa{Zq9KQGEI7YcFnfu@f;98O@{tG!P-VZg%R2gT%Y8?OPY1Z{*?r zc|BkJGx&kN#kF{A^Uy;+IK}^_?;P|l0V1L&l_$NRG%}j4*1Q;hP-`JkNP!@|AgjW> z^gXuJX?51#IJdsF7J5NyjgL|xq$UMxnOOkle3kd(ZrZmu2tr6fWXLjF6cQaYaR)E# zW+i(*AWDH+Ct6z}qzj`I(6GOAhj4=jF94vQCmKLhvHt-#V$8ZiJ^}!U{xD_tNfrPG zEx0MFjEsWG7~$Z5vrT*cDI*HOD_@r3BlkT1Ll6CbertoadV0O7+dZ(FP0D1DERZGZ zpl%?5h{*w1`WfeMU4Y0WWa4kg5A+OKq>CK_q~Ix<~NYY9WjD#XL);l z4Mb3^l&v)gS_nvElbV=}=XpvfFDR5`zSxZxmGzWI%vNiEgTUk_3m|)e3Y5QCz-XoF`B@c{ z5n$PBW(G2UKmzephCtfn3$dvYz=>dFQCK{1#8Wg6#nNTwMxv^~J zh)|ht06t`t{jA!VTaPzqGgXm-f3bxB18DW!vw z2^0v@Rl(d&I5A)*B>ApwAe z6iEUHLNZuQNJ1c_Qurh@-d(5jU;fk8LJ0S~bNq#7r?cH;Yr$$D!$eNg-~gdJFkAGk zL&=bTnS=^w?)$NC`Z4}S>~w=t$zQ7<`R;EfV@ync{*lc?)&6HK#7=Wz`PR*LoJv3A zb1n#cZF|&H9a@t{YpqSfR%m0<^RZAVw_2@kD`}*(wzXXj{fSZr3EHG=t{TNFy>8bF z0L(JBU^v`h zcn7;$SqA5i5dqoB8GxJa-gma8U?64?1mrBAW-D0%5`iI-CL!_Ocb2-%?%6Laa=TO6 zR}rZttr3aII+8DB#y%j8*>R>|AE$!5a-DzvTT_cl1@4P~ zzxh98=IyPOcDo0VORNt35WQ^R`+(AzsGAr7!(iDG8z#1ZWGq?D0ze44u-#s1C1j{n z9mzo4*xp7a(nf0=QA%VHgpksc9wI8~xnkL{Hw*g&WVX@l0#yJA&;kYmSnWhs0DXUk z2R`DCmKHe36z#rC0v;qDaS+Mlz)7-y0Dvf&oDL&qjZA0|r643}2_-a>;M$jGX=B49 zz5UuY8;z{l#$X8<0WGsd*QtpPDMtVUetO2g=+w)W%fKRQG&;>FYE# z#E8O43?d38ncV=cJ{MFUX*kFc#<3>XBgW+Vo%^f{F8Bg)7;E=W`f|m<$bxsBx!W;4 z&M|_WPEE$X83T5{kN_aZ?d7Z>39v|{>QjZqo6Y8O8={UT3q-`)882kUezWjSk=Tt| z#{RD$Wd8pHSUsWwhHq-}7ywOw8gFlICdMMLQp#9%;|MjgVf2CwMiJSBtQM@mSi#m) z7!pd}5@yaUr1H*9LaJr4kgZf-mvLOv4+6htYJKqdqn zmNEv_FE^O5zirr;BcuV+pdX|4o0@k#k-qDI01Osf4=f@x4Pdqc82j*lH6msqq}RH- z61~=y)__jII^yaGf%+eUKA>a(+n`}N(p6UmIPSmN%c3X0xdLJU1Y^=PPMk)EzH=gg zNeJ`35Szjtk#4MaUt$yNvhc+Dc%vEChH47qO6hc{R`<~RUBSJK2)3f$s^72sVvP z015!0WI%|HSq;#z00}tM`YX3y0{|f0JqP?YsO!M%?;jEDN0YaI?GX=R#KDi5u%MJm zb?hF_fTgQHZjXp?@D0v(Gqb}y005Bz02WBjf!lR4Uys54gMhY&3I)`nBsTaojW3GO!CN(IvAbg`v7NcPsFM(c1Xo zg9rETA4@eU&j%}9Z=2)yeoM!7=r+XBKny|{zE{+2P&oToBtE8C50dsuI~sfk`TRClnF&I zi8i)wUZ0(R=R5B`dUVX_E*Q-WQoa>w_pXa!W)R3g)OU)gFDec!aGOikM}QA zFZBIjWo6~swO88h*3Fx<+uK_gE}q|5-zZng^?KcZ_r1ICzWY1A<2$COr@Gy))&>#D zX)+<4@^UlCoyqQJ6J2E5*Gz+8ckmd8kN2B`#PRkYTxF8b9r?bkaFqexxhdZHTK7-f zdrmpJd1PW}XKwds`=`;#2m>PzE&}^cw9jWJn820K)u7X-FFgaOvg>&u7tT3#x9eVb zaC`lK$=l3;X#0f*hqOf>*?;|wYwfO9GBA!O-VzIuGC8kGNGYT=Ow#9Wk_y0FHXlXN z>PGX($vrmhg-Tf(gm~Xz4a+IR$Lz47D3FK6R!5spJN*N$aSoNtXR~v&^I!hbm!E$6 z>E-1mAw;*+Ri5(wj8gvU^7``PswMl}vtRpvlRx>~CqD6iyXT&JYqeUZ)3r7gLI4cl zE<8AIymKU-A31z_J4$gNMnl1R3)w-(c)_lpdMFLp7fTqZ92{nnxQ|{H{c-?+fSmwj z5FraD`^{ok1nht3z~vp7e|K3lM0PVHf*MK>Gw%Yr_oJ>i3F**78YrIrnIm8+Jj+&p zvzEdm5JCjYE890_7laCpMGy!A9Y2)|tx!sIg?hr&XrZJ4nHWatYa$DRQF@+2!`2Q` zn20DONEVFFUD-L%{r~RNW%~JZ1mHgOqmL&5BqHC>pup|M##f&D%5VMVCs&qNgcO$1 zXzK+bGaE~m7=)BcDHgyycWdsafBK()O-)XH=Z8M{UElRxVVH@csQF_`pMRBuy#3QgLbXK4)XwY@m3$+B?$dm4-enww>pe(X;=rp5C^Yz8@8P+ zE;ID;KyNZ+Z+S~QMon=4EWn`Bhs zaybALh#-{kbdqRoT&|F)&pjysjj<^D2cUlzZrA_p*x80Z?sz(4M`Hq7Yl9$MTwMI6 zU;5?w`J0hb6JMqp-v0-&GLHP#SW-&X*DLZFcD)Ly-Db7N~uda@Vy3?#%F z$Rd-2;7i~^q13Roz$5^r5K@2u)~O~UCHtKNF3@mgnW0gKK`aZy@(TigFuL5h6R*L+ z;Qo_jwnU!qw_A<>{eSZO5VU@4>!!jfhC4S(T(?(Y>6n3*gKgj^<*rfD2SsZOOQjp9tEWUQTESTNS)@;MbG ziB1rd07XpJrp`}iW+KZBAj5vSlq~b07mkg~VW5!xJc1B?-&D|TCX9Ev7kyW<4{?OP zZ$f{{DzYy8+L746ahuFvIvs642X9?$b~ zxtx^J_x&VEeBa;L*w9*kBi*H#`o(S_!~rgDU}bk@3i}?R+d**7oJO2mSX^JNKYrra@gr}4`$N56 zPl~>O;jZeNcmjRels|XD^}oY%7ZZlz-o1Md9Xd2KGjsa%>BYsx`T6;$pMH9FcGeX) zzW$~ZFp_2GU+vf_8fLh6EEh8gfeb7RS_12dTPtg4&s~(lQ&xB+y_lk=S>M=9(=>{V z)>ir+8MXv}X&gD^A^0*VU;&q^<{)YrJ*((<>eDXJc!&>e8UL0!#!Nvfi6aoY?u=2b#HO?{g zWSC;2ZHg+ej1q-R41=~(10z^rBb^DopZnRLZMWKgk3ar_Uep#sx#1+lj4*Jk2W6n4 z8|1dZGJ*IXLWYL>J~K1({`bE>ilS<@die0+lP6EU@WKnmm{zN`wY9Z7ejKj86Fm&9 zV25Ss8{V6X-r_jEcJ(?ENa^~K(E}#X&9=5O5gT?)dwI8%AP#?dsKE`uD%M zw!Ve{9*UG8ANmfi5O@ldx7BP*;)3suR%^{xvymi12|>cuQV<~%D9{!FnaOD5c^(k~ zs9*k1{}I4E{`iM_QD>Kt8o+|F4;&bX^}s8C#c!yk{(CNWNnsd1_~3&?^q%*;CyJu8 zXU`H*ynb{35{02O|%!I-ZGiYoar&20oSW0Qx zl3|u?P0E12?^ki^N?o`12?*!De-VlL;3q3G61d1FmE zU)Vb}d1-bz>ULPF@yf{LSpCvX+ejL$0aywU0!bp4G8xi_k)p(QlPFGD2}F-PRKM~| z{|T7i^S<||X)+LB!PaFT+;vF+cKjrNG)Rg6kKhBj+|Ah7*vCHhv8AP@H{N*TZ~yJT zU9DDYwHgs^ZfmssV1OIz|MX3MB>^X?4ujB_K|4;Wl^T;k zUy!!iq!w67kGSwxl#;h@-Foi1=Q2S^#;B}UDrXLim21UZFN*fg zOl5-L>h;+`p8QmO?500GY`b=wT2U25ZwKRZ1x-{>$%t z^1z`(d-v|q+CV?o=Mpgt%qYcwFu1wBO0eTOIQIzptVp+I9sgVCs-ga_R4OAQBV%J@ zANarr&YU^J%!{(mBy@s)r7jvbJA?bytswQhLxg)r9@Dd0*@9niob2 zGZPclViZO7a(=bh>ViRIJRyh} z0g!Ay@DxJE^T=AIAhFhelaeg4l zqtUR|?%TKT;>C+%hYENgKtsuLj8tz*I=|(T$Uv5mj_lv-p?vOt#aCvxdzDIYde7ck zwd{HIT&~dSbW*L8)DlygrVKI+Lm|)@lO%DRL~$IoTFq{^V|$>jjZK0`e&9#I;; zDVOcUX;WavQxfhPAfa?IuYw+_~Mm_!pKZr!Pi-M@Juf?6Fp>Rjbu%wVIRH zD2i^~x^>Sz_xQe#i1*%mZ!VX6_St7IUcC7A-=hQ$lF`Hf0$2>}7bEVpJ@n-X5!?)F zV6-5VII@4w)Y#Y?7cO1BvGCFhFA-wBJ~BBuDG++G&SV2(VXQ?I?RJNOtu@x_BuTp6 zPPfz5CKcL$L5HkPNt;w52m!0Bw%ypuWdo_=t?kzO`qqJidn=U^5lSTm((u%hK<1qy z7xudvcQVzU@~&UIy1B8L4MVb=>fUIrnnT*$*sPSxQb<59mkQ$}HIwRcHtP$pS_cX$ znM~JoZJjAW1f358u$=W|p_s>J)JRi*Llz7G!~DYh=H_;8CJ*d*OW1js zz^>_gAOZdC2m-_C=;+?Pd*A);ci(Zx9TzWNY_(c@_wHrpOP4Nv`qQ62efsp|+=_r9vIYBKKjv*e(-}IbgFJ@YN}K!0l?PQ*1>}ZM@L7K zBw1a5U9H#a*4i)(;jQGg59e?I>fN0N2#&hEUuPKz5CfoK0y07b8?kk5Uu+9yQxt;i+7qpp=h@M2YA_O@@wt!VvAz2TwUTB!WniboS;=2~tSU^ZZak zUU^5Rr!IH9o*(Snx9`^ct=dR6x3n}_E0wYt6YHE8mUG!sw%3ljf$;ObPjOuERVANE zBJ!2zi$n=1=E5j8QJQKZLpBJ)?d|6L{KDZwN0K;ohyZp4Q5Z&rZuk1wVP^I`@AT<^ z)4uOtzkdC}2OlgH3O8@w92prYm&@;X$2%4m7oT|IiA$F*t*@_7PEI0X5Ck`F+*nvx zh~wD#uW#Daq3LiOU3WZ{{};bPW|<*-@5l(*Bztv}y;rvE%*QOUvO;#;PG(j?b{+K-X9uJVxDyR?U{=JzPSg1gyBh4L zRSRKM%i7u(RL!iLpn!n3jt-Ng_ZN7{55!1)|H#!-)fhM{1fwa$(hXPHlKhduAC@8# zMw|9WFL^&r1x)UH%kk-JASRK%GK{jLhPIEZ`Cr?_t=CpkkCac|eF_(BsT_0iEp)Rf z`kcVPC}=Mo!;Gteh}I4G^ZR!kya0Y55rjlleIDLsR~y>&iTH`;?fO}C2isWb^v+EE zpH^#5^Z^z1D)jQ@OGNTdm#&>zsbI|B;;41i2Q*8!rdSqEiIY-PPT)CgWV8J17>Tox~{IyeZJ}R6gs^;$Dq;Z^CoCxq{JBMyV84K zX=Y-AJ57BzV7x)Vto~a~jqm0dl3gto-haf%DL5nC4ug^LAhMm?dexF8iaE0S5)u>V zfA|j@2ktgv?5ijLM2-Z>R-j}9kGGQ8RJ8MCXV<-(_Q5a{y4tF$Drav2SSW4k%gN>E z**3fo4p;`V5FQC{dBNN%tf92t(DiTYT_GWVrd|gt3LrZV9hhyaX0M{UCjSZCyJ0ph zjr(7Km}QV%>#N#LK<$c%!c_Tr65|%tvl&p}%X^kcd;z(6`^0QNs zzy=xr9PvXU0X3GpTA-xE&BLzV_0!E^)Xl?R<3&;GyMUzRPh!HXoqFmJZsr4hu6FKq z0`_T*TRMlvpr17}4wmzq>=X0Dc!7kpQWUwt*LW8J-7)o;(>SL4Ip`CS!WV!m!&Mwz zQ|@!it^`AWoxu~Pj)2d^6>C*bSyFCAKN^7NW2Lhou%;^%pHEYw{we*SO>Y- z@HtOuLWJPS56rDt1q`hj{kk&GH^FSX3FLQ^Q=mua$s-RCv3@PqeZs$mlohAf(iZg$BJzEY6LFdY7KXGw$8{723+!0X654ZGze0*$d77y2lRT3G47sBxF(o4KL z+IWF{#0tW-+}T=T-f8bk-n4fbtExPho!RqpzeaEp1=3p|2CYiXKbDbctgO`KO#^{m zMeb}c`1}|=*lIlPGv7S`Rs>VQwTn~OyT-;oNR>{yh^2Ia*B86f_ynRPq@)_YfB%vS zO>G;5@!4%%`IEA;vgT9Xb>8;8J%hJP3R7+bvawO=9S6mT|MbesU0tLvd#g3PHd_?Q zhnG%HdVfcAi4#1E)G3Rc?@?JM1WZJk=p*am}*DPjo4l0z?L?_-BD6j zd=}dHb`U^vWdbw&U0hsTT_50w|4F(V&YaSfE#YAu5Kuqr1oqOn)tVsnsi~--yNuEj zLPC2o^9M5Nf);Kl<3!oBHMT_Qub^nGto+x>miWCsfaVQ|OH8bcav?vAks(K=7Qz31 z;fZ8wmX4|qe>1PJ`2Olo7gruV6VuBTaQz<=XFkqD6R`k7nVOnfB(qlpE$y{E^~f2y z>i>*#yPs|A#G+NW&&~3EDjed01|}pOQg4&J3FEl)JJU2b%Hm6W$oe|<)xT&CHjcnGWq~iuDY&G|>P9C8xPp%o z0%9{%3F&{nbCvu3_m6a=nN?wRn*hv93tthIU)4^8y;zKiqet*xLP%u=_VoAi^78RH63IJ_^GZ0s1?CTodklPEtklDXr5{SiaHl52c~xR} zLnD5|RdK|uIJb_It$hx~WlvRfLjK%uW$m;xL9Xp#WX}d>i}SFmn>=SZV-xg2%B3QpB8Q{wi@`hBO zG2^QlBp4^ewrCT0cYEI4r02E*x2~EI_fjZ6LWxMeTmlb%@{yHQi6xwQ#KQ&KIgTJm zNBzEvkO*B3kCpmuEYoH&{4Prs-l!kD2%Z9P{WR5hd_GFXOl}`;d5W7M?#&kX6?%iP zJU%vNj09=+B{~$Eab@;rC- zQI-~_3h+&H-#u(RJO)cr06^-p&8VAolN$Ydezn|_e!T~%*K`51GdgI|UHW*!2Ha`b zzr$gj>(0cBPOzN9T}Mmn0sqZcTzovj(J9V7)l(dMuJM2R1*erbi?l8|6GD_~nQ zg+A*2N?b}Gv6QWZ^Oc9$?&~kXt05=^w80-$-*e^nhoAkSTNN~uIOy&*yjrS?P!x@y zO{Q@ve~O1m;1)FCKecjr4|>vnW&AhrUcv`Gdrrcj4R@tMaAnX$;@D#)B+!uhi}NHc zu37yJabL=B8WDU(j*kUJyOedU90X#v-OG?Xp)Q;=B9C5VW~@2$vZ{XD@tN+Udg$Ee znCxVOWGVUavsmICTP98uiE5S5GbbBY6A!^hv{4Ux>&p1;V-Z1ye5)Tf8T2y*mu2R6 zpFXqrR+8t&sEXbpG`RMm?*L-c!4D_YM$iBl@I`Br8+>e3tXBl33eYAlo`C29iq4Z8 z7*7r7P$PfyXMMQ1kX!M0TG~ApC8T~509`nOY8+={C z{lo;@#A!fa;`e?qr2B3+w%^j1b~j}HfD(+m`ediX&Z8T!zh0aN?d#-qQMPVtQz(65 zOCb<3s4&YbPe)sr8k#)A%shIK>YgI1=WV&GI%2b4I2tonqNYXOX3led)gk+8X>D-Z zJ{IS-dNQ^GGXd$hLQ`Fd&M&BhWs37gydi{Melx_iwLvFO+)|G-2gZ&&;}fT%mydVn z`7VEcuA&s4a^auLHkmZkK*lV78+VM#?eu5Nm`LX@id+0L={)YlCZk0+`x-HsIqz2H z`t44*^>I<}%ge$*kKGehnY@OYA#1E5l4q*&=>uH1iGrPdvS4>~!coVuU^wbqxr;wR zvdz$sAJQ^1GD1RsoUG2lT9M^6gZkN>>A7RES}5;giL17@mMP@q4}c3IFZ+XztMeK? z7L~=aWC9KWGG4nWhW7UMz}e!bLZSp(CT3=&{dP=c^X6;aEvgJn&CHIZb$A|$i(>!? z2K&}e-U4d?I5#=C(x8;WK-U=Pyy^Nkg-*~sCO8k}3%l-?Y`Hv$VI2(yxza_<5ZlrwKEQ z*$f{cSHw)Fr>W-p<|YjLMAIKTIZofwA5(P7$S6vV`Mk2_D-v>A4h8l-No$133SJWz+RmvlPe{NkqB9*BH=m&AJ@HK**G<=?8RJyQX9trp_CATm-r z(jvO}RdAS>s|72>r%d`*j#qq1UyIljF8}09+dl_Qx|95bBhRLsj?4Qh%*~7@U3^TQ zO%3zQWFBz{ZMp?nLB4}MIIJl`W9kJ#>my%lG%+64PT!wr@yqfX9PsfIGDCmYH_(?(0n%P zR(n2&X~LZBEyBA2>bPnVylA<3<$1C8X;U70we_K9GGy!J^<4e2lvEaaRa{)$EP;+J zkwMCLZGeU4_xwc`bTIic?@>Dd8%!r*OebA=7hP#-X-Fgz+#v&t;72K4CrxtV;&`yq z(o%SLZR55p*gX*(JbP+;Dg8c{29X*}H!6-V@S(X+1E{K1lw$Fp((|CEWb`A+LEFY6 z+GmhBLP99YrQm%}uB^w>-_wJv&CT?30YJu~aku!K3A{hvQfJX~5`8DCrtsxt3viR3 zd;7yc9GjiLoG5v&@^?oJVN5V8ZV^q~S|mcGl-k{gJsB@Vv+yxHc3VuBFf}@kJDmBO zON!`Gq`TUDz4z(9^Q!&oD!@)+S*0u0ki9qiI~yIf+i~R6YwH6Y8FqZXN_)HvsMAF| zJ$;J!2h+c15M3~f+GbcemsZWaBX7}MaAuvy_-e|)7V=+uhZcI37hwxM*4nMP2JG|= zyw3FZ@$s>?wzjwDeD_Jacu{dd8}z8xEhb3afD6Ikb&clphUW9VMCgL)5-Nm5Wn|2_ zDYKc)K@|n!%~h-1MQc=46iM*f6XVU*JSfD`alIvq2++ZO_SO2O2*(YV$(v&LU=@Ac z1Dy)qoFWUJWocv{n6{Dm@#DM269&Maw6`l44tA~{fU0h8YAOJ5cozQ4Y2e;#>FA6* zq4Z0alk27YA!ba=U0u9>&uu67>0jyoodv5FftOP&`_?QLTN#vv=94LGZJy(VIh@Gt zJ98o)=*D7DP5h{@8FW(Krle6|e*g46Jb&l|RvF8e$Xq?)cPZ5UKYSO*F61t%m?ZtZ z^Mu&R&g4y4V7RUSZUISqQMie*=sLR%^A8j8rxcNvh`dC`VTJC|4(*iVk;hdu_zxUSEQ*gkkNv%xvbT>~?5 zC6)mb$xS&8N+@M4eBmjAlF^%uX5{@_Kc-7mb#>(5e{s zvKoXsJEPA5E&nzf#}v>|>CkC<^J-wz>MC@#3W&UqpFu{?)Z|R>$&v8*`!fvtdc)$9 z&!A$!I1!*ie*Oz&^HD8;ji%TOGc(yBiU z;J{2a;|#7ZDjyL7OC`s9RW)`W<9%Q#_uRcK{*Z+u`6^l*z~R*l=a_8Kq9y4 zs)-sU?8&) z*X@nex5cXZ8Ocs&lh@;cb7tJjVIwK%rWzr?qYCKXUONDtr|ruJbS6lmYCem=MIT(> zuNkv*6TNBOcz2ngj%h{-(Pzy8NBahXrjCAPC^bq1gL=ByG(5WXoHH584F^Pv1Lq;( zo1mq4>yOL>#14)jzt@>@+ILt9DI-E&Qm55vOvPx!e)5F+He3Q$BM;&v!SY7N9thuqa3)5R~Q zpsUl>2@f6X@XM^GFLCW;(B83fxAGGx{$jL3@H%ZGfDW zJ1igpQP2fc%l`hS0Lg>ZVk(|>Gp7fuL{TlkNHG6;F&>G0K%F$Y>BF-OI7W)(LY2g( z{cb)*{70#2RyD6GDO%M9C6%$fb&s#ge7}A8%gt-+C2Or81AFZg;#M_Fh@F>NL~aWq zt2jq}U4qnkuEEDNnp-2E_Qb^7xa@vX;>Ayr3wctUtgNgk#Xq2Y2NBNP z$f)l6>Kq6)Pp?48JrAiEPl1p|QvS)byb^o!`irAC+6>%VxR9PMO+k?tYW*r;9q-;_tJg7^S|l!z(NVy5oL z+^#A{-BM7xh0ENa`!QNx9p_NmJvexncGE6M<>_|0lKMFL4)KRqohiS3-eJkmVmbnd z)85C5aOfG|Q5bBZ=!jTtGVR6nW0LZ{I8=kwD(fWz(K#4;>45!}7O+RYmdr#wnfZB{ zzEd|kA#p^%+-;5fKPtjyO`3pQ%VQWX>Jyx`)uSo{EA6EL4$9oWowP1z4q{7NpWsH@HH{MUI^RX-WG&Tdy2Jl0ssUV5`Cer#$%>$4>>vU zrgZ6*j@}?iaF|$u%x)}@ZlJG!yfX`KpWJa(;`V&=JVMAB_0g+pb!8=6L9{*SM)?T2 zI2js(3#2;(8zMc(D1bJ*;nxQpyd`sjWI=rqFJw?*{(zEj(#6-#j)S<>x52Dp9GxR& zz)cB0B$*xDy>2Lt͸L3<-7=hAbpWx%i!yM|AWb03$5D^UaUX@`IX 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 z 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
cHmOGqT-gM8iKlIC*#_ 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

qeU&HeX6X_*AfSP1O3=m8Ce=uLlebdP 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ÞVXi>eMMq<8T^AAHXwnZ~VKexTRJ4Lo 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 delta 698387 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

%enc`te@v=yJDy#fy)7 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

Ar)Iejj`X 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

|U+SwM z1@)@QE^2F3S}_5|OtEDmWmD;>UY22?47Vny14P+sq(ZfI%V?ai#0Xh3fHldSQ8G7` zo`n#9K*YXhv_?QcAWHxsrEImLv9YmC=CYn1(a~w*I5x)A>eWK7(B2kmvesHD#rW7n ztJTWqvWN(doQzo_z+7P>qvOPPb{1P;B47c6*?G)h8AK+}WNG>!mnKD94`#DmnMTi} zW(}2xy-R5C7M>WPE%yR5Ob+wT&0@ty-n*558a2*!i1u!^@p5_gvqAaXsF>+#&T(>T-o2Q)jOiN0040DC%t)uXe@yu0Q$D(E|H@yT>#A*-ek{f5pcB zsfy-~mTEPN?dgW*Dro{xI5U;C8i^Z!Q6dJ~MJfma>7jjmqHc^89y1v!d_pu<3+W+C zKp>|G06?Hr0>~)k(Hwm=3nBLJ88U`EUk1W&E9M%Fh7edR=Au?C5tZiy@XYKi0JwmU zv&k)46at;o&M@Dg&CKNTP9OvdiIRw10SX8KaZ6+hRGGD-dqF5nrNzKXdtkSJkOI3G zLy*RL)mwG0Sb2_Iu@085)!TnTJxk=u4YF@3Yt5wf@GbyU-HSygYcoaR2O?ikO47_| z^9Nr9vVdYPN5Kq109u1JP%qcN_(>?VVZMV)6PV7a4VUV^e!J^Y$`oL1$TaIy1Xn)5|+EI_2jK-mzuZ)oZ%Q=_1aEC>6lm z>ga@%1%d!{gl1%k4FdsxTQXWFEnw|9^KE&?Opc%*$o^$=@GRNZMaGD>9@)_i*`j&) zPOevI=yCFV(`q#D{$cYc-y4VUyg0(VRz+4ePik^PCXsIl81eaW; z(pi|Da;xhcTLt1=4%OAQ2?>oFPmMqlbG4*@{-f5H|8Mg6BQUam8H4&_Ki#~a#-`)g9Gs}- zQ9U+RzPKygrGk{P?fG;urSekfdNSJ}6@cI38yPtR`NF~WNyS@+L5~T!0AQcD!uBg`qoh8nF>SW6p z8rlKU1A(A_6aa&_gf827V5U}oZ^m8S9YfWIUo4K7&A`}ncVEx< z_tsuAFg?oYX^@c;aVj*~qRr;84Yp5~!hBe&8p*6YiAs5_p`JucB!sUdS8B~Du|i2^ z6hZ=%WneNyD1_%`DKIk;3G_h#cx<&142Z6#kz;H2>>R%4s>|w@2+{MEQo`Cf7FOZO znaSCIYPqVU01&S9(GkZ0AdqDu7RY3Q&=Om#1q!wXki8&8*RBSxrozaMKB~^r_;6gC z6=52C21F3j)Il2AgKb?jHUyr}Q=_=<{PcD2v5lr}walXrQE8klLA}BZ@`9_?;?=Uf z8*D^6L1r(P5i+g!?U++JPCl6Dq(@1x0K`Cl7NnR{QBLl|6j%#HAOQfu55W&`@*wa2 z1qLA}k>380t(Kuy(wpzJcYb#v4dkkllZe*!^^MI=E1OIPQ^Lp6zUd{4@&_ks z-6?O&#MJry+2L3(Dah$rn_|+&OgAD{hLUF7Rt$w_(s@d22%*2A{%hOx;IidpfTl-( zvuVuNYQ3rS!6@m0cyLKaJ=6A+T_YP-52U>OqPA3jHYg%=Seuu$DC;*{tB4!51O-4jynURqXF)PmHJe{v3rEn_}0PWbZ zKWZgHB7z8lfQV5D05HY`fj=`n-KaNzgp_Xaa}g#-*NQ|8EC_@JMnGh;76lSB0s|AE zU^`bh!SyQFW@vU&_Aa)iNwf7nZ|wyj70rxut4^h98r_dw{hTZ4$A3W+qyF`8mhC<4 zg<|zaC{J>E+Kvv>z5f|M{*dfjB3Ev}Fkl%l3VmQN9oPj*f{;Ki=!~E>C<#t~^}3QE zW5M%4dB6-r;LS~zfg}cEPzn&JIvbCUKoUc(OhY?y`Fh&*Xz}{j_^t|o_ycGT@)$~-dDk~7UvDXoGgrpb!Y zk%Nscn2mY+Xg%)*!&Av3p%J~h=f@K@EmMoTvyR40kOhGrF_UF>%)N{nEfcdPW30>l z60C$^**B(Pv|ju$Z#A~1oCiM0?Yn1GlPL)Ok^>WRI@%emEn>MUCWGozx|3E6^u z{2`7b?Cj_L+d*qKS_Ua)5-?gn^i#8EDTKiQI}@!JvVZ_!X11by4-9pHNoD+u@26VLwxqmgqSR+~y;ROL>-AZ! zf?(&t#JQ zU4_|nKvDvrASMP@%K0_qY>%i#WZcYO48u$qWQ0)Qc$%4+T=~*m=Y}zc$dV-nww3^Z ziIh@PQ&ao)9tZ+|kHH>ZLt=@HF+;;cT5G3r<{r8^0V8lB#ksFAF1BQhF~+(K%G_xH z0031>qGyR*y^*5^=DSb|u9fWML7JK11G_Ov)a!2&ZC!M5ubml(Fa$3kLnK3rVhmCU zY=?KTHfr&5(Yr){=d-M*e@hIb~L?*czec0Ce5_|pXzFaQ9N3m+V9Hmj|P$u(-qr{4-So`b=qqMVAm~R(e!05P!P=u-A|9lLiDiC2h6pNOD zkWlP@@UFgIEna1YcZpUbzWWCdM-YTS7UF2Gpv)BjxmFq>!IO^K0i*;(kP;XmNdOUq zfFwya-wkA;G-YQdj;wzejWv%?#>=Fo47A5;ukyP((p$DqEL+;An-K)67&Wpi>va>w z4M{jWo-E6!_l}p>X2kwVvsH?xcO1xecSdD@t(B)jj5Ji5(ekqv5B3xvJviFg+tF#D zcS)hIX8g{gH(52x-@UKV-7K%acGb5Y7`o={u3h7$4a+;a(%G%Em11R9WzuWgQumWe z=l!v|U)3!D5J2Q+)4RtVmYLSer0eWMr~` z#>PVU00_)mw?1C2w4_vXJ!W=J!6b>EnwnNhxso3yW2|KoZlpJHLKL}#Kd`fx=dvA6 z?V;ng24G^5N&}*P@Mm`CW-=C1c^KOdafCAy9JTVVd$&Z*ag6O<`p4gh{`q6};X5#C zs6xA!Z=W2ie zKuC~cP8B&%k_fB;Yk(P~0u-Rl+;YY=7sw_Hl^I}yYB~PVw~n}cA;KNI$0w@htn~wX zc1^YXpgLA6#ip}S#f(&`U?NBX+5NGcw#;Fwkxk1mfcCr>c)Ad#f-s%261ly9y|}cH z>hI6DWktR*(*ew0n9R;3lMQIL;;AasV_T_lt(9uD8rAZ$F17sZ)dOkh?(*~9#m*&* z>ydoXy3W}~qj#|T!LjCeGnv{ulaeZB4Em~-kV-5GsaOyr0kKf3J~~yKi4c(a(DV$U zgcPI=8$tkNaEpilB+^7E1PYXYEF~gnod}hYD16`Bv3=Lh9sAGSu*zD^YymuLEdn%~ z(d6W`kdiGUqA@M93Cj@J5^ER#AtnT1L?Q%Zm=g;|f&#E?5v7{b^PP*JQL&!S!=v1& zq91VYQXbyH58WY_uMIAKj-IVICnm7isn%bddd+*i!BwEZM!Z*KgveWraW!)&DyL8(-1vAI$oJ5r+ONl9d_0?)hg|qO!_-J6B*88 zGf9UP-;WW;d~X(Udovnsw!APrd-aMfJ4SoUX)-}psQ%@>kN$jfcVT&YDq3;&qV-*w z^wO^8%v4=wJt&1KUkOwwI3AhuRKPx4W26u>v*k)N@`NO^04z{{0?bw00b3>w>@x^A z(So#QCL$nV1)xBmtVv`g@N-hgTC;KYo%fx0-r6{B8A>u~C4}@mEX|gpR*Wtkz@&8~ zP`D~$6bLMs8JU6El4Y_$mdP>`f}%O+-ALsDpdBAF`yYqu3=}$Gbf0b3#hR>KzD_M) z=Use_t(B5DZ@>et*-4z?IPXNmU(>7*x5Z@4?R3wUfflG7QtjH*gxIO6vCRuQq#7j zWE-<7hazP6}pQQ|4Ya=CnD zJMI}gCHVS(xBf+rCL4xQ-hzu2|*x&0ErS12oQw_K!D7DfC7~e2rxHu08ubAAOR>uuuK4q zC<1-yyD1ew5 zgkbyl)~ye1dT7_)A#LJDt7R-@LZ8@LmRgfk3J8)UjuH=%$wt2N8S(z z_Fwga=k+h?H`W3IIi3v^WC`$4O^y5E#K=qx0#N{?Jlz5$0F_V+@iaQcYfp-c@2LTQ zqL3IY%+rpcwarqfCrc0l8j3s9>)O+28BR{u`3oWE>cJV15#~rz5Ev~O z!4d%F+^qu!;mZ1~CC~R}CdU5r-@mu*@qNM*LSZY?LU=AVtBn=HBfxAbv?d{c22Bj~t82lssEYhQc*%U(LTYPlg-cFIV=$O6zwtqU`gV2MbUQAlxm zmoR}?H_^`KsE^*01xNt4C}~sKmFeE&9_8^=f)Pbxs?&`}l?aaX00>RZjMt5LHgJ5p z7BWT`Aw{X2#p4VdDUdB82*A02h{N1KE@mLh%2(qf!{7b-f9xF^7GV%Y)+K+mj)?#u z5C|Yx2u}%AevoKl(gc=_1{R1)3A9E_PXIz38JP-3M~A-ut#7^Xr7s^`y_}E*2+LLq zWf_48-KZ_Y#H}2yY<)B4BlVLxi3#AW9`t3kVQ0Frqqt{y`+i2M>is zBmjYIP)cNHN?VBnQOvnSCVvk&wc=DmX$iC>2U%wnj)q7MiRQ*hr$3AIwaQ|Kz{^^ZLS#g{}_ERsbZFI0K&< zk(lNzAOj32!B}(pZ|5O@!hwZezv6J6L`Qrv1D$T^0l>M+Tn2C+mSszKm$g&k{CXh;4YokPi#Ib|>c0ImF`aZic-h608B{Q&P>+(x~mdR?HfVI!K@lQ9% z01D}f9gvwA-M|!ATj#1Ukp%-XNTo*i?fb@;zFlrMr0qR>fk4FAl9ZmW1Q7`IjnrIarf=JTm~b<*k>$apFU{Xhs7QVF3Lh?$X5pp*yp@U zS$uc?%BO>*bAhPzMl#`KSY5-rLwtn=-c*Uds|0dO6ch5GypTRB_a-0 zDw|Gc(qYQ?r4k?kSqPMj${p62*k}_)5d$$uOLBI0YH06%0M>ELc&O;YSQJ1^NCeh0 zu>}xtx{c_6XK@0#cRuht6 z^i(QTK@fI!b=At{#Y>k1>1w%}%jShpp;XfMnCwisM6qebt+{R%PF%sTu|~MIbRrN$ z#9Eu}TefUGf1^Z1%NCgkQ6OqVM%zQ(Tuj7p`r<%;|0{5u2J-2&qX__r1qwvPYygQZ zGlGx^z(@v!WCCEb`Mw7xW-8K?ewYsZP&3N_a*{KFQYy%0GZL`9vzM$fI_~J`jpG(u zlqf7yvsM+T*d#8jV=Xa(5CW}00mv4KwN5}N0St({c0Rsm*Pc~tR_HhZfgn&2lC?ZX z+y}&ebn56(M*#@wm!?LxJhF{ZTEZ|0z)yK7pjF9d3Q~%WLR(u~yU__UFcX1{mb@ z(@<+60TRMLvFk%d2%Ib3g}Jm70SN{sH!wsi-+OG!w#lig2s|kRKg_GZpW44W9eNFa zFId#uyK2?>%u(EG0o!_`8bz_S+9p(Ycn-9Rg0S1S^x$Xb0>&GFgojs8H7a3 zHg=V_zPDq?&Y^uH%U3PY#sUIa1ORkxH^y3Pg;LCiW#bvxNAKsAoqJmofjB}W&~5X) z_Kh80Re8HYVP=ecO^{1FofsUF68HzNrQo@z^9&+p9npdUl(0F);~Q!%BtS$-sM<4r4<9Mr%fL#6h*H~z3oRb$@F#qI zh*_d!1_r|{7*HU>oINBg8wpTO`q*Hk;eEdvlU#VA0jx8|$RrXiA}^@&e`gzDRT{YQ)wWfKbZNN@3zU5$QyeO@tK0 zl8iwmW~Rnkjh2`4!5YV_VXYB};Ns)PfPGRukfT`CqDFgL|FRWLV9w+UqZ7l;T8kONFz9ISLdj9os@E!z zSORV~8nGpf2BfExsM;uh1CWxUYMBI-O4VAu-WC*?EDC9@aeQCg&mGV zMe+-ZDIT`jY5Gbg1{Q)5JYX?bLk1QIfRUL%;LP-t1{R>0wFEFpk|}mrYdk;YdnzSD z2}(*4c!5%aJk{5?B;VHA-Z4$_yp- zYNJ-K&7}NvN~KeO*<K6Qqc!-mHdxkPgz)+KH)2ZFQm0Q{k!56LD;e!l$1-AP|^+p(aKS?A<$x99?BV6kQiy zU}=C8u`R#?YO;iPgz%Ahs!ax52;jr9cv2RDoTs}Sh`J+}+ST89$AFsmfJm=Q}I z%PgWdHzBMYW6hIoa<-;7p=^@U`t{2^UQO-mk5QItHp(?qypWlO0k%*97$n@btaf-p zPU-*-tNyI11gFDRRwgQsOvWT!^uG#?{DE1dC9P#E+gPK)Vm&@NTdif`qi!pTk@!rO z<$bn=h=e3tN*)%k=?Xi*3h}Iip{Q@xL>PY6uBYP=IQ}QOUqwcnWIo}vn(l> ziT3`!8fISEtFu`$)LaaSYC&~YI+s_2+`aDI6v@`mt#`A+E(G{r{^O?JWkUYM5S=u` zz=QnN+6Ehj*Q+nbk1QTsP3o;Wpm#yFeXBm_6x>e_c;;_=thJ*KY(y-X+0~2_q0xIf zi$KCc$)53Y!w7*?TPB*{mmTu#$d2oUN>eio6yz`B=$@^-)tc=naumBoThHi@te>Pv z#HisNh<)Zff72V3>2#U&b}zl$bUab1c*Yh&*CCGr37cM;TK?!*W~fv+U+aRO{%*I~ zo{v#%{QDcq!4Gxj&&Xw8_^YCkMRU+A(q?jySy!%L-fk!YXFRI~SmDh2;vEGxD{$YX zDhF%gk8odph1jEI7-V;#AGeB0EshF(E!-EWHC5JJunV)wnB}==z9dEddN`Cev^_EB zE1V(x>(~2KWdnbUg>08sZ>(M|e5Z88e(vui}<2R zKW-di@#a$i;M#WJc;emQJDOiJB=8G96TxSH&a9D6G~$>ynN_;) ziilZkyn+<0&Z0t{pd(J(HkR4sF!OFiEfFUSE_&bvP`~v-d#+<7xRXxSiHyuJB1TJY zX^spbO3RoTs8ab7`)|;{mO{X*z>pNeDpi2VscA`9R#NI=em(XkjKu=yV~%3JXwVnq zT;!5ME$W82PcH*6R>ndTzpgq2d0!q`eNa_5(onF95WveB$#8dM2EFizSYt)`2NYhIST4M}Q3W_g2V!XNepha;fLTcZyG#I{?$5 zs$xWWt~9eW)m6SVljwsEhvN?Gl*fjKAL)6u-tACQ^LzwPLMayBo~ECXzLc0l%7{8R z=5oV<$ZaI`I^jd{k@#kx(ai0$oF2`#v{7DgazqJd-7KP zPQ$$;>zZq|@c)$9f257bIbZL8p0qWnp2MrMm-jlj)+u33%*_xH78DhIllyaWPc*Jf zDTQd(b7XQ-O?Se`kQ{0y!5Wr=VEM8^TSp>CBtwv}?k$?hxW8jjS1`|m6(Q;%c&3^q z4Pf{693MU}Zm7)SX&ew%DEVg`Za&&cTAbDy$tWYx*^s3aPj7we0sUz_;jt@mq;1AoC)RRmEAyslcOVjf$cXYQqC99shAZ-%|2up!AKv* z2C8>uQGs$Y(!xm@|s z+0iPQ(s_x(DrXd+fM?xwGo@qX>sAfzlr#A>WKdJrc>dDvjdn?$XI+8qDCA29`7$8J z*HX&Mho5RY+a2K~Ilp9Jw}+5X!ye{5qntvLnTU!7 zPtiJ7nt>0moOPlGLKyjwu=QN6ox_m55es}qKc2lD!cJC9ZTiQE_%6R5;3HA(eU93? zjmesV6%NN0$06^wS^v$BeH zcuhp2uv_O)AS&5Vn$RCGhGwx+%$JKrW|WJd*Jycjd2gBIR%F;`J%*#Jm=~HD4GC51 z8RbJh<9{DTt_YH%m<_Z=%o3+0eu`(7IPXE_qV8q$tp)&_)E(TGlv^T&07ZB#;l)}Z zEAOFl)YGcQ_zNeA|DvC31uZU8VmDlHCrBy_$`de`ktNQ3JcyX2?52p#{A!T8m-6k! zYbuV7PY2ihv@q_fSOEja;$jse8b8m)2FKf*#=v3vDJ`u2DCC$)$Qp>)pF6A}#L*2? zHlk-QF^pR-e)eTLWGHVo0VG^VgQ-=pN&FkE5q)!YUu4-(f>f?>F4k&`_6BdY@Ghq+ z>^V_?(Z+YJ2v(wYyPy1;`Ot5?4W1TXUQf)Ai7uOHl!Wnjq?9Mb7OEl{+CZy6;NeGm zv6UqlIPUo&86l(5LXG+%X?aR-#qaSj;w;ER@li!-a5{|IR`(OX0{0Ozg6UW~w(3?k z*-0GANI2F6D>yY(F69FLG8r1mnQBETT}$OUc>cjyPL{Kxl57epEJ1Av{UKadof>}Z z=>8ZYQ$&`oV?QDrsN%60!{v5we~>eG>pLncBRM5A=#E+CT7?rKUuWuck3UC)GPyPA zKrQf5QNrF1kYGuC0{nTU<^8nY&hh242>g{#)|^YI9Y`});YVSEtd?qoC8Q{lV`)sp zXhPg9BT#>Rf0iZ34VCU5#u~4+NGfpLt}{2puKpgiK_TKD5iKn?vn9422?))2u%ek2_efMM!3Tw|960WrAVS z|rvTr+wrMVaAJq z%A1FDQT#{pnhx4{gUdZ-XQfCeo6WyI$C2v5wt3sCK8J^~i5^-!)6EnO8rUPaGI;h@ z9?wlMI*--66zf_gNP(%)i8}OiIMw~f#j8?FykM8Ew!C{~eVz;oy$qwQP3f3AhwiZh8JRX?q;NKnSNYl^`-S-xfA zdkV3vhs}?WuWa5^iUEUN<3DuEdJSpbHcT|YE-`6rtX<8;RsJ)Bu{KIhW zpYs!6i)3yNPbAt1rlAxOhFVjF!2ylZ&pZ5VhvuSL4WEBz1ikenY-YiSWQ*uxO1e`5 zz1N6$aSB+y=fu1PRX>}}`^NqQ_mufK-3;LZNybM%Nlkt3pLIHrgl2tBOCXM>U40+x zRq%%WH67WR7D>`-=KF>0#j3~A!=+SeYU)_|TH^GJL*Hi2TyS52u*-kDF~4|F7<79? zd6f!{H0xUEUp0yg6o^kQfjg24kwQaK05=aQq!21hD5N9PZ9N`*>k+Z}jJVvXNB47} zQw&yvI^m-BEX2I}CH#QwsDNdNbl(5gumGL;S9B507X%7QVz-_ z!zoaptEbn~p7bB`LW*>X3e`aZ z_rh!f-(G7?rj~?NU}bu;rPnyE1BE3@2H1nTMD=vZU97|dX-s?PcC#Nhqnv(9r^zHS zTHAcpAf&CN)KP}Z4HHdD9hl2@A*ZTz8_B`o=JE{Ho3jr2&NZ4@IVH%1HC2Y%hJ?2y zL{)0}joXng!U($MN97tMCB=+O-+YdhCz|Y!btoW79Ni66n3*zgPHll{1K@(@7_D^D zPDbq1)C47Q4u6rr=bzgF?0T2P^H}oC<^746NSv6`-B?5{P<(tNR?T+4#GPVFIUESc zrIB#*w{`FmEmHn!7oaQH6x*iKIcLRr*DFQI^TCvs36zuZ?CJfiorZTNW+n!tb7r`E z_35HJCT50y8}|D=H;4TKXyV=|n`ZpOmc;}2@RL!-8ZJ2SI2H&%S4FmS(v_}15TJ!M zh}4FWK^<8cd1z5E(afvOcX9AyrQQ>UStQ6%2)@9UU(F4OPC~*IP9Cl>Y!Y=Z^Rp{rWT8TKMi>m9?jGhu zcWpW-$M#IgQpxZXxfl{9+GIu&tuNqVG)Ni&GG0$yIu6mFS;Wwtnjy1 zoJ?qIU9d{r7&-ulZeo1AQIjCe!xSwhlWJ}{impUP-A}VrP*Pwv?REX;PyDc$RLL&U z=bxxMM~AWbzhiJ><66WGr?$ks>!YUcImk7OHx}|@Z zViqjtHlUM%J5pbZ629k=|H2?NPo>D(Hj@@HQN-XsLNNlM1r0mMPJM^P^vxA^z({a& zTP!zUS^NO|s|R8^X_cuSW`BNBs?ev(@4z-mVuuWmMpG@)RFANS_nDVP=)9bs950Ej zTVX0|!V_tHp~D)|N0(1ok7JIMNZu!Dk16)^V2eRq3$_w8Je_&U;yXtvP zXQ1XAnq&gi_>DY6u5p~F?@zB7zD`oIu0F^;Qsmu~_%Y)-50jrpENZ=|vXRv#E-gQw1BwDFZH+O|!=uOjkal-xScZq3+z#lHpyn}D?+7)bj0`cYI- zp{W=jYyr{vK;vk)ujcVZzDN@M)2UUz?or(~6h#$Rsaci{><`kgU(j3*-I*gveCkq~ z+B2F3&QJdARg>c*g>VRa^T=PCq?N%L`WY~s78K3ICHQk)V^RBuGc(Ds3b#?-C$Xv< zF7iqn`fKs<8u2%dW>7ooSokwD@Zh}`x=6F>QPHIK*sY7p#}cQW7Cs9J?Io)_Xla=Z zjQg3W%!7hd0Cz;?<%G=uwQvd$xQ^ZAz?BvRn|3vUf zL4%v1(momPYZClbDO#MI50(6Cr^b+j`NQW*w_iu!OUba$;{M_iQ6+XgxRJ$?Tv0_6 z{!!0%w~s}_4nZD)@P(OsI;nLw<|)H<8LR2p7D2PukDOMuOms1g^NVIskP2$DPOFv- zU`W7GgvFAs^J86>@79L-n->*3jF5RwUixJOhC3BISEVi`Z%U_T+e}2pVbQaaAt2SWJnStkT8ITnZbd9_6uz1VPvKB1&)aW3t|oBH|jE0 zxZDV(?PJU7|MFUER$L_|SW-af6Bexz*^VKB5M`Y~JYQ}xk%p6i665RagxW5rUsZ6X zxcSE0aJbdaGibh?wY3dxAMY15s2l@cS4=N5agCJ{fehYCGTo@~<8bu4@Hku;5Hda+ zm>n<9a1mpY8rcosTe0w_H;!;LqxU8x)!-$hM02L7IZzoI4-h1OmaB|jGI2GbDCT3$ zcfdoL*k$pTw4|M$mO}|;Zn`wxX`l6sk;pMBc6403U{bjc3yM}i|5~-%4*zDhIVlVZ zqvJNB6B_=@xSkO47F$(yAch44KqAl7Y$U0BOm(??WZ>qjv_QinXcw@7ywAey{lHNL zzP42?cejn<sOXpJet7Pl+enQ5b_V z6{k4g2ik?U-NgIUF+Z-~cAncUB}jJ}?{6<+ zsKrO7mZql1C&zPC(6KUOrOhpAshFSf@#P{t`;`zH_~F_1pVn=L>wqDCsgcA>E#6;EqhF|U(1I?|Wf_AZm;Rgo4 z#PitWsAj}O8<*`E6BrUsf~GKQ>OMqg^>%|#K@J~o`RBq(!7^6M-q9+Hj0h`3!*_th z;0v4V;fGs^ReB^}Rf@ZDH{_lb zgb0=zw$unFjhRs_z>=Z^8Ndj^>)~WfEf$x(*uHdbKs@v?-Xs|a74!67aLjS@-f6Um zPc61drW$R0GM&xFsJRZEH2E*?zTi&)Q_lzXK{P^49gl}{C5|4L_?dQ@JiYc#s!Z1h ztq2)sPxLBX*z?i95I#)2{VkNQa^b)6aSteIM1&0nwj0GffP?>3MUx472QK)v7ewMp zUbOjkDe@zC7xJBuh5pu-2(>!8q(Uu@pxFtJSDzwgG*VYao6xl#%XBEJdb@qNc-t+n zP}+LJ;^N@^+L-y{dneoFwbMv>pvtGmq+u;TaC^=3Tj{qsxq`5Wy8VwXFVE5x6#6%J zZEEBNrPVZ`t-#+;?LwLaA9CoDqdf@<=YPd!ofA~T(BPd7crWnz8drY~^u}Q1%GrG8 zoU~CH93hPeY%FUT8BIJ+ecHwoM!NQ$6cZ9p@y|-KCdUAyhlR+~_DX{w8T}V@i(QqdsbFn&1>ab^d9ab9LavBp>@K$uXa#q-zc8$HguSk=?(EdFOoQ-P}Gqa!A zOBjVIo>d*foRAl={=!JJ=gFlePw8<{kB zCA&X_0gbjj)hXdWxk?!#7k{=eyzrQ#W#F$!ZpS!snx&=xmO}la1fP)-x&a+_c*sC6 z9K)O(Gde*-}r&u7}Ehs##LM8vMYWc?58>v*~~SZf#?RXzJt>tD9VP~YY|g@sR<5qJtd zK5>NV6L62$4{e=zYne*p5D0iWk`c1tHEojja8``?y&ku_yDQ*0wQ+il{=yfgl=SKo z3Al&QuNA>w6b765?Y`dVwKpVryAO19bX?QJ)>uu4%^YCNqHQ_T7c+D!)NeLgw}4;mhFQbg z(`80JS6R#)@ScfEg|?)g!hER?O80!^jYA^*pbdJO>!V1i!1Y+6heO1};q`W2+sK6` z(ZPJZ>u|1wQN2shMxw$Czl}ujHz;sPuzixz@=aQO^V{=(lMuKZGqbEm&4-Kzt&+PX z#Qjoz+r2X@bq7A}gO-#8(2~ywJXbEsa%tSXM*xp^a{y@h!#X48a_i>8s}?8vAg?z3l$ z9>WrO_xrX&&T~W`R@qIDKT2IEOBvMI3=-ODuT1g zI;fDMef>FIXyu4rMaWp}O_fE+beI&Z6T{F_6gfXXC`TKHuPModv}4|WYQE4T_H&WE zNshy_ybhS$slt$Tx0a9_dH9zd}{5z#mw+u<>Bvb-&red#Nk1-Az?oT-=lUqSLD@o6aEvRvMct@{DFx1L`q5pw8|6s^-ouSoyJJyV}S zf$8RTYQ)vjW$WeKbZc6a>GfZ|-0wK|C-se}>`DY>UPP~AhI79y1*280mTboUZcT z)}FmD9R7>}c{n$FtR1xrJS(6%4+54x185+iR@&Ty5CjTt1t~ZpVZ?!UB!+y>QydE(w2Fy7V>ZaEI%A* z=iN-?JznSiNz8r2T2w2TsvLCu$!!1NAZR5N{o$DA(cScA{9aewqmZ~o?K|xduwzJ( zmU~hPLtkLjX+}K|{9cub@6qg12MG>(fiL49&mKtsm`rA7X5J#&9{;jehzF*+hk13N zk_O(b(>#vG2Tv?5J%2RO&@DFxGS!MJ8n!6(Vagmu{M|UupD(%nc@}jQ>l(NoZhSLd zX(`q=%IP-g08ZNP?UkMSWzJPnQ{PpZhgF)^vm!2Meu|+iH&Vy-?pVjEpGrv^PcPBfe+W)ZF?q&n@-Z!O80g7dWO6kr3bDM_WQlVL&>$`mlvlSmlDH9wGNZb zLP1?Yi}}YtZ6!(p4HkXJwwuX{7mo%xFY;C~OS4Y5z^fIEM&KXitfXsIrsYJe _K{!+W7MsfO zI1i_|E&eL8KljCE@8Ka14vy2pZ+yse^&l!j+q!IeB_N4GX-)WfSo>v=Pd-=&X&#Ou z(VtW8AP_>`%@5}lj}Mn^Qi6gPb@eR=MO;DlWFH_S^Yh?gUIdCkBim_nHt1y>=L_9}< zG>A`5Elo~!JjhJFZTCV9MW9!gvj;~(J4-7^mut_(GcRh^EGnms#_@6w8>4fkY7N+EB{uqh&Vcm?3HE`Lx~O>j;tKu z#7_a-6azQs6@WzE@G!P=neJ`Q@^uav|CEc1i;?JpaqHM2Moy(^_Mxd^BZ$keR7nMe z#)!B;6Rx{Ed|wHyjYOiE*2fDhTkZh2CxGiAMW)gh4ZePJ)Mh|IiZ6yU-c3y#mqMxg2 zpVxNtH|Uxil$xIZcuGVNk=`*Si)0Q=Yu-^yMoeuq(75TGz>Ke;Y8}`fq-8ZVe5uM{ zX3o2|1^xGtxR@Pvw{ZL67V%gkb;~EGTBOx>oz;ffV`AugqYb>ox!<@6y}n!&#$+F+^p>de%sw9 zSDxqnG~r5|He0IcW#jaJ?l=&xfZ6^uF041mMA1j zd^lhg+pKrsaw!0z9(+P5Y^|HezmOx-1!nxYn9m_jSV-nFn2{aP!j{F?OTq#mr}zxA zngg?klXMW_>l;%d_)heG>K}v2`&e5M#L2hpvmEpU=tDVTPiSD~yBhIyUCLOXJ634P zcdx9zQS?ZaDS3E1o*a4$K-Dns9}tfZ4coaYa!;@mBqMUl{65Nfe;d4kWe}EIblAYD z2eB}CFG=O4JAwugopVMmPJYO�snn1+PDaP!glc!x8SITVE36(1KBG;5Fsa(TB%V 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

D64}a)Ge=Ex?ON&djI5O7W+P?POg;&~p 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ⅅ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⁢IR$Z$lx!*{6} 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

oScwJd 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�M(R(ks@->BgvAnRzby(~h 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