diff --git a/.github/citools/cpp/clang-check b/.github/citools/cpp/clang-check index 4982137a..0856915a 100755 --- a/.github/citools/cpp/clang-check +++ b/.github/citools/cpp/clang-check @@ -2,6 +2,6 @@ shopt -s nullglob -echo clang-check-16 ./*.cpp ./*.h -- --std=c++17 -x c++ -time clang-check-16 ./*.cpp ./*.h -- --std=c++17 -x c++ +echo clang-check-16 --analyze "${@}" ./*.cpp ./*.hpp ./*.h +time clang-check-16 --analyze "${@}" ./*.cpp ./*.hpp ./*.h printf "\n" diff --git a/.github/citools/cpp/clang-tidy b/.github/citools/cpp/clang-tidy index 9ae9dd14..eb362d2e 100755 --- a/.github/citools/cpp/clang-tidy +++ b/.github/citools/cpp/clang-tidy @@ -2,8 +2,6 @@ shopt -s nullglob -CLANG_TIDY_CHECKS="*,-llvm-header-guard,-llvmlibc-restrict-system-libc-headers" - -echo Running: clang-tidy-16 ./*.cpp ./*.h -checks="${CLANG_TIDY_CHECKS}" -- --std=c++17 -x c++ -time clang-tidy-16 ./*.cpp ./*.h -checks="${CLANG_TIDY_CHECKS}" -- --std=c++17 -x c++ +echo Running: clang-tidy-16 "${@}" ./*.cpp ./*.hpp ./*.h +time clang-tidy-16 "${@}" ./*.cpp ./*.hpp ./*.h printf "\n" diff --git a/cpp/.clang-tidy b/cpp/.clang-tidy index 66b18073..518716a6 100644 --- a/cpp/.clang-tidy +++ b/cpp/.clang-tidy @@ -1,9 +1,9 @@ --- -Checks: 'clang-diagnostic-*,clang-analyzer-*,cppcoreguidelines-*,modernize-*,-modernize-use-trailing-return-type' +Checks: '-*,clang-diagnostic-*,clang-analyzer-*,cppcoreguidelines-*,readability-*,modernize-*,bugprone-*,misc-*,-modernize-use-trailing-return-type,-cppcoreguidelines-avoid-magic-numbers,-readability-magic-numbers' WarningsAsErrors: true HeaderFilterRegex: '' AnalyzeTemporaryDtors: false -FormatStyle: google +FormatStyle: file CheckOptions: - key: cert-dcl16-c.NewSuffixes value: 'L;LL;LU;LLU' @@ -34,4 +34,3 @@ CheckOptions: - key: modernize-use-nullptr.NullMacros value: 'NULL' ... - diff --git a/cpp/.gitignore b/cpp/.gitignore index a018ada6..05176049 100644 --- a/cpp/.gitignore +++ b/cpp/.gitignore @@ -2,3 +2,4 @@ tests.out memcheck.out .ccls-cache/ build/ +.cache/ diff --git a/cpp/README.md b/cpp/README.md index 17bb489a..44110d14 100644 --- a/cpp/README.md +++ b/cpp/README.md @@ -14,3 +14,5 @@ - [hello-world](./hello-world/README.md) - [lasagna](./lasagna/README.md) +- [pacman-rules](./pacman-rules/README.md) +- [last-will](./last-will/README.md) diff --git a/cpp/hello-world/Makefile b/cpp/hello-world/Makefile index 6cc59662..62b4b657 100644 --- a/cpp/hello-world/Makefile +++ b/cpp/hello-world/Makefile @@ -9,12 +9,15 @@ test: clean @printf "\n" @# each line is executed in a subshell, we have to daisy chain them so they @# run in the build directory - cd ./build; LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" cmake -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi + cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi .PHONY: coverage coverage: test @printf "\n" - find . -regextype posix-egrep -regex "^.*(test|CompilerId).*[.](gcda|gcno)$$" -print -delete + find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$$" -print -delete + + @printf "\n" + find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$$" @printf "\n" gcovr --print-summary diff --git a/cpp/hello-world/compile_commands.json b/cpp/hello-world/compile_commands.json new file mode 100644 index 00000000..036d01e8 --- /dev/null +++ b/cpp/hello-world/compile_commands.json @@ -0,0 +1,17 @@ +[ +{ + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -std=c++17 -o CMakeFiles/hello-world.dir/hello_world_test.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world_test.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world_test.cpp" +}, +{ + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -std=c++17 -o CMakeFiles/hello-world.dir/hello_world.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp" +}, +{ + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -std=c++17 -o CMakeFiles/hello-world.dir/test/tests-main.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/test/tests-main.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/test/tests-main.cpp" +} +] \ No newline at end of file diff --git a/cpp/hello-world/compile_flags.txt b/cpp/hello-world/compile_flags.txt index 76a7ddaf..5dfc4cfe 100644 --- a/cpp/hello-world/compile_flags.txt +++ b/cpp/hello-world/compile_flags.txt @@ -1,2 +1,8 @@ -std=c++17 -stdlib=libstdc++ +-g +-Wall +-Wextra +-Wpedantic +-Werror +-Wno-unused-parameter diff --git a/cpp/hello-world/hello_world.plist b/cpp/hello-world/hello_world.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/hello-world/hello_world.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/hello-world/hello_world_test.plist b/cpp/hello-world/hello_world_test.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/hello-world/hello_world_test.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/hello-world/run-tests-c.txt b/cpp/hello-world/run-tests-cpp.txt similarity index 53% rename from cpp/hello-world/run-tests-c.txt rename to cpp/hello-world/run-tests-cpp.txt index 53ce9df0..ddaef161 100644 --- a/cpp/hello-world/run-tests-c.txt +++ b/cpp/hello-world/run-tests-cpp.txt @@ -6,9 +6,13 @@ Running automated test file(s): Running: make clean rm -rf ./build -real 0m0.015s +real 0m0.004s user 0m0.002s -sys 0m0.013s +sys 0m0.002s + +=============================================================================== + +export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" =============================================================================== @@ -17,7 +21,7 @@ rm -rf ./build mkdir -pv ./build mkdir: created directory './build' -cd ./build; LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" cmake -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi +cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi -- The CXX compiler identification is GNU 10.2.1 -- Detecting CXX compiler ABI info -- Detecting CXX compiler ABI info - done @@ -51,11 +55,15 @@ make[1]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/hello === All Tests Passed === -find . -regextype posix-egrep -regex "^.*(test|CompilerId).*[.](gcda|gcno)$" -print -delete +find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$" -print -delete ./build/CMakeFiles/3.22.2/CompilerIdCXX/CMakeCXXCompilerId.gcno ./build/CMakeFiles/hello-world.dir/test/tests-main.cpp.gcno ./build/CMakeFiles/hello-world.dir/test/tests-main.cpp.gcda + +find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$" ./build/CMakeFiles/hello-world.dir/hello_world_test.cpp.gcno +./build/CMakeFiles/hello-world.dir/hello_world.cpp.gcno +./build/CMakeFiles/hello-world.dir/hello_world.cpp.gcda ./build/CMakeFiles/hello-world.dir/hello_world_test.cpp.gcda gcovr --print-summary @@ -66,23 +74,25 @@ Directory: . File Lines Exec Cover Missing ------------------------------------------------------------------------------ hello_world.cpp 1 1 100% +hello_world_test.cpp 3 3 100% +test/catch.hpp 34 27 79% 1642-1643,1710-1711,2227,2229,2560 ------------------------------------------------------------------------------ -TOTAL 1 1 100% +TOTAL 38 31 81% ------------------------------------------------------------------------------ -lines: 100.0% (1 out of 1) -branches: 50.0% (1 out of 2) +lines: 81.6% (31 out of 38) +branches: 28.6% (8 out of 28) -real 0m8.911s -user 0m8.253s -sys 0m0.655s +real 0m9.165s +user 0m8.313s +sys 0m0.844s =============================================================================== cmake-format --in-place CMakeLists.txt -real 0m0.159s -user 0m0.111s -sys 0m0.048s +real 0m0.224s +user 0m0.178s +sys 0m0.047s =============================================================================== @@ -104,94 +114,64 @@ found lint: Convention: 4 -real 0m0.147s -user 0m0.110s -sys 0m0.038s +real 0m0.198s +user 0m0.139s +sys 0m0.059s =============================================================================== Running: clang-format-16 -style=file -i ./hello_world.cpp ./hello_world_test.cpp ./hello_world.h -real 0m0.024s -user 0m0.010s -sys 0m0.014s +real 0m0.020s +user 0m0.009s +sys 0m0.011s =============================================================================== Running: ../../.github/citools/cpp/clang-check -clang-check-16 ./hello_world.cpp ./hello_world_test.cpp ./hello_world.h -- --std=c++17 -x c++ +clang-check-16 --analyze ./hello_world.cpp ./hello_world_test.cpp ./hello_world.h +error: argument unused during compilation: '-stdlib=libstdc++' [-Werror,-Wunused-command-line-argument] +error: invalid argument '-std=c++17' not allowed with 'C' +/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h:8:10: fatal error: 'string' file not found +#include + ^~~~~~~~ +1 error generated. +Error while processing /home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h. -real 0m0.540s -user 0m0.509s -sys 0m0.031s +real 0m0.544s +user 0m0.491s +sys 0m0.053s -real 0m0.542s -user 0m0.510s -sys 0m0.032s +real 0m0.546s +user 0m0.492s +sys 0m0.053s =============================================================================== Running: ../../.github/citools/cpp/clang-tidy | head -n 100 -33313 warnings generated. -143486 warnings generated. -176792 warnings generated. -Suppressed 176789 warnings (176779 in non-user code, 10 NOLINT). +11382 warnings generated. +46771 warnings generated. +46771 warnings and 3 errors generated. +Error while processing /home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h. +Suppressed 46776 warnings (46771 in non-user code, 5 NOLINT). Use -header-filter=.* to display errors from all non-system headers. Use -system-headers to display errors from system headers as well. - -real 0m13.057s -user 0m12.988s -sys 0m0.068s -Running: clang-tidy-16 ./hello_world.cpp ./hello_world_test.cpp ./hello_world.h -checks=*,-llvm-header-guard,-llvmlibc-restrict-system-libc-headers -- --std=c++17 -x c++ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp:3:1: warning: do not use namespace using-directives; use using-declarations instead [google-build-using-namespace] -using namespace std; -^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp:3:17: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -using namespace std; - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp:5:11: warning: '__llvm_libc' needs to be the outermost namespace [llvmlibc-implementation-in-namespace] -namespace hello_world { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp:7:8: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -string hello() { return "Hello, World!"; } -~~~~~~ ^ -auto -> string -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.cpp:7:25: warning: calling a function that uses a default argument is disallowed [fuchsia-default-arguments-calls] -string hello() { return "Hello, World!"; } - ^ -/usr/lib/gcc/x86_64-linux-gnu/10/../../../../include/c++/10/bits/basic_string.h:525:39: note: default parameter was declared here - basic_string(const _CharT* __s, const _Alloc& __a = _Alloc()) - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h:12:11: warning: '__llvm_libc' needs to be the outermost namespace [llvmlibc-implementation-in-namespace] -namespace hello_world { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h:18:13: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -std::string hello(); -~~~~~~~~~~~ ^ -auto -> std::string -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world_test.cpp:14:13: warning: 'hello' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - REQUIRE(hello_world::hello() == "Hello, World!"); - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h:18:13: note: resolves to this declaration -std::string hello(); - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world_test.cpp:14:34: warning: 'operator==, std::allocator>' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - REQUIRE(hello_world::hello() == "Hello, World!"); - ^ -/usr/lib/gcc/x86_64-linux-gnu/10/../../../../include/c++/10/bits/basic_string.h:6185:5: note: resolves to this declaration - operator==(const basic_string<_CharT, _Traits, _Alloc>& __lhs, - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world_test.cpp:14:34: warning: 'operator==' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - REQUIRE(hello_world::hello() == "Hello, World!"); - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/test/catch.hpp:2342:14: note: resolves to this declaration - auto operator == ( RhsT const& rhs ) -> BinaryExpr const { - ^ - - -real 0m13.059s -user 0m12.991s -sys 0m0.068s +Found compiler error(s). + +real 0m6.209s +user 0m6.139s +sys 0m0.069s +Running: clang-tidy-16 ./hello_world.cpp ./hello_world_test.cpp ./hello_world.h +error: invalid argument '-std=c++17' not allowed with 'C' [clang-diagnostic-error] +error: argument unused during compilation: '-stdlib=libstdc++' [clang-diagnostic-unused-command-line-argument] +/home/vpayno/git_vpayno/exercism-workspace/cpp/hello-world/hello_world.h:8:10: error: 'string' file not found [clang-diagnostic-error] +#include + ^~~~~~~~ + + +real 0m6.212s +user 0m6.140s +sys 0m0.073s =============================================================================== diff --git a/cpp/lasagna/Makefile b/cpp/lasagna/Makefile index 5b106230..62b4b657 100644 --- a/cpp/lasagna/Makefile +++ b/cpp/lasagna/Makefile @@ -16,5 +16,8 @@ coverage: test @printf "\n" find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$$" -print -delete + @printf "\n" + find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$$" + @printf "\n" gcovr --print-summary diff --git a/cpp/lasagna/compile_commands.json b/cpp/lasagna/compile_commands.json new file mode 100644 index 00000000..9d1a33c9 --- /dev/null +++ b/cpp/lasagna/compile_commands.json @@ -0,0 +1,12 @@ +[ +{ + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -Wno-unused-parameter -std=c++17 -o CMakeFiles/lasagna.dir/lasagna_test.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp" +}, +{ + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -Wno-unused-parameter -std=c++17 -o CMakeFiles/lasagna.dir/test/tests-main.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/test/tests-main.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/test/tests-main.cpp" +} +] \ No newline at end of file diff --git a/cpp/lasagna/compile_flags.txt b/cpp/lasagna/compile_flags.txt index 76a7ddaf..5dfc4cfe 100644 --- a/cpp/lasagna/compile_flags.txt +++ b/cpp/lasagna/compile_flags.txt @@ -1,2 +1,8 @@ -std=c++17 -stdlib=libstdc++ +-g +-Wall +-Wextra +-Wpedantic +-Werror +-Wno-unused-parameter diff --git a/cpp/lasagna/lasagna.plist b/cpp/lasagna/lasagna.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/lasagna/lasagna.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/lasagna/lasagna_test.plist b/cpp/lasagna/lasagna_test.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/lasagna/lasagna_test.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/lasagna/run-tests-c.txt b/cpp/lasagna/run-tests-c.txt deleted file mode 100644 index ec917368..00000000 --- a/cpp/lasagna/run-tests-c.txt +++ /dev/null @@ -1,285 +0,0 @@ -Running automated test file(s): - - -=============================================================================== - -Running: make clean -rm -rf ./build - -real 0m0.008s -user 0m0.001s -sys 0m0.006s - -=============================================================================== - -export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" - -=============================================================================== - -Running: make coverage | ansifilter -rm -rf ./build -mkdir -pv ./build -mkdir: created directory './build' - -cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi --- The CXX compiler identification is GNU 10.2.1 --- Detecting CXX compiler ABI info --- Detecting CXX compiler ABI info - done --- Check for working CXX compiler: /usr/bin/c++ - skipped --- Detecting CXX compile features --- Detecting CXX compile features - done --- Configuring done --- Generating done --- Build files have been written to: /home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build -make[1]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[2]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -[ 33%] Building CXX object CMakeFiles/lasagna.dir/lasagna_test.cpp.o -[ 66%] Building CXX object CMakeFiles/lasagna.dir/test/tests-main.cpp.o -[100%] Linking CXX executable lasagna -make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -[100%] Built target lasagna -make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' - -~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -lasagna is a Catch v2.13.6 host application. -Run with -? for options - -------------------------------------------------------------------------------- -Correct for six layers -------------------------------------------------------------------------------- -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:18 -............................................................................... - -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:24: FAILED: - REQUIRE( expected == actual ) -with expansion: - 12 == 120 - -------------------------------------------------------------------------------- -Correct for 11 layers -------------------------------------------------------------------------------- -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:27 -............................................................................... - -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:33: FAILED: - REQUIRE( expected == actual ) -with expansion: - 22 == 220 - -------------------------------------------------------------------------------- -Fresh in the oven, 12 layers! -------------------------------------------------------------------------------- -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:54 -............................................................................... - -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:61: FAILED: - REQUIRE( expected == actual ) -with expansion: - 22 == 220 - -------------------------------------------------------------------------------- -One minute left, 5 layers! -------------------------------------------------------------------------------- -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:64 -............................................................................... - -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:71: FAILED: - REQUIRE( expected == actual ) -with expansion: - 49 == 139 - -=============================================================================== -test cases: 7 | 3 passed | 4 failed -assertions: 7 | 3 passed | 4 failed -make[3]: *** [CMakeFiles/test_lasagna.dir/build.make:70: CMakeFiles/test_lasagna] Error 4 - -make[2]: *** [CMakeFiles/Makefile2:111: CMakeFiles/test_lasagna.dir/all] Error 2 -make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[1]: *** [Makefile:91: all] Error 2 -make[2]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' -make[1]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' - -make: *** [Makefile:11: test] Error 1 -=== Test Failure === - -real 0m8.642s -user 0m7.882s -sys 0m0.729s - -=============================================================================== - -cmake-format --in-place CMakeLists.txt - -real 0m0.178s -user 0m0.116s -sys 0m0.063s - -=============================================================================== - -cmake-lint CMakeLists.txt -CMakeLists.txt -============== -CMakeLists.txt:15,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` -CMakeLists.txt:17,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` -CMakeLists.txt:52,71: [C0307] Bad indentation: - ) - ^----BodyNode: 1:0->IfBlockNode: 46:0->BodyNode: 46:52->StatementNode: 47:2->TreeNode: 52:71 - -CMakeLists.txt:63,70: [C0307] Bad indentation: - ) - ^----BodyNode: 1:0->IfBlockNode: 61:0->BodyNode: 61:11->StatementNode: 62:2->TreeNode: 63:70 - -CMakeLists.txt:68,02: [C0113] Missing COMMENT in statement which allows it - -Summary -======= -files scanned: 1 -found lint: - Convention: 5 - - -real 0m0.146s -user 0m0.110s -sys 0m0.038s - -=============================================================================== - -Running: clang-format-16 -style=file -i ./lasagna.cpp ./lasagna_test.cpp - -real 0m0.022s -user 0m0.010s -sys 0m0.011s - -=============================================================================== - -Running: ../../.github/citools/cpp/clang-check -clang-check-16 ./lasagna.cpp ./lasagna_test.cpp -- --std=c++17 -x c++ - -real 0m0.361s -user 0m0.328s -sys 0m0.033s - - -real 0m0.363s -user 0m0.328s -sys 0m0.035s - -=============================================================================== - -Running: ../../.github/citools/cpp/clang-tidy | head -n 100 -16 warnings generated. -110168 warnings generated. -Running: clang-tidy-16 ./lasagna.cpp ./lasagna_test.cpp -checks=*,-llvm-header-guard,-llvmlibc-restrict-system-libc-headers -- --std=c++17 -x c++ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:1:1: warning: replace macro with enum [modernize-macro-to-enum] -#define OVEN_TIME (40) -^~~~~~~~ - = , -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:1:9: warning: macro 'OVEN_TIME' defines an integral constant; prefer an enum instead [modernize-macro-to-enum] -#define OVEN_TIME (40) - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:2:9: warning: macro 'TIME_PER_LAYER' defines an integral constant; prefer an enum instead [modernize-macro-to-enum] -#define TIME_PER_LAYER (20) - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:6:5: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -int ovenTime() { return OVEN_TIME; } - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:6:5: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -int ovenTime() { return OVEN_TIME; } -~~~ ^ -auto -> int -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:11:5: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -int remainingOvenTime(int actualMinutesInOven) { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:11:5: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -int remainingOvenTime(int actualMinutesInOven) { -~~~ ^ -auto -> int -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:12:5: warning: missing username/bug in TODO [google-readability-todo] - // TODO: Calculate and return the remaining in the oven based on the time - ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ - // TODO(vpayno): Calculate and return the remaining in the oven based on the time -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:14:12: warning: 'ovenTime' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - return ovenTime() - actualMinutesInOven; - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:6:5: note: resolves to this declaration -int ovenTime() { return OVEN_TIME; } - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:20:5: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -int preparationTime(int numberOfLayers) { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:20:5: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -int preparationTime(int numberOfLayers) { -~~~ ^ -auto -> int -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:21:5: warning: missing username/bug in TODO [google-readability-todo] - // TODO: Calculate and return the preparation time with the - ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ - // TODO(vpayno): Calculate and return the preparation time with the -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:28:5: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -int elapsedTime(int numberOfLayers, int actualMinutesInOven) { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:28:5: warning: use a trailing return type for this function [modernize-use-trailing-return-type] -int elapsedTime(int numberOfLayers, int actualMinutesInOven) { -~~~ ^ -auto -> int -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:29:5: warning: missing username/bug in TODO [google-readability-todo] - // TODO: Calculate and return the total time so far. - ^Suppressed 110153 warnings (110143 in non-user code, 10 NOLINT). -Use -header-filter=.* to display errors from all non-system headers. Use -system-headers to display errors from system headers as well. - -real 0m8.600s -user 0m8.540s -sys 0m0.060s -~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ - // TODO(vpayno): Calculate and return the total time so far. -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:30:12: warning: 'preparationTime' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - return preparationTime(numberOfLayers) + actualMinutesInOven; - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:20:5: note: resolves to this declaration -int preparationTime(int numberOfLayers) { - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:1:11: warning: suspicious #include of file with '.cpp' extension [bugprone-suspicious-include] -#include "lasagna.cpp" - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:8:1: warning: do not use namespace using-directives; use using-declarations instead [google-build-using-namespace] -using namespace std; -^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:8:17: warning: declaration must be declared within the '__llvm_libc' namespace [llvmlibc-implementation-in-namespace] -using namespace std; - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:11:5: warning: variable 'actual' of type 'int' can be declared 'const' [misc-const-correctness] - int actual = 40; - ^ - const -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:11:18: warning: 40 is a magic number; consider replacing it with a named constant [cppcoreguidelines-avoid-magic-numbers,readability-magic-numbers] - int actual = 40; - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:12:5: warning: variable 'expected' of type 'int' can be declared 'const' [misc-const-correctness] - int expected = ovenTime(); - ^ - const -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:12:20: warning: 'ovenTime' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - int expected = ovenTime(); - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:6:5: note: resolves to this declaration -int ovenTime() { return OVEN_TIME; } - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:14:22: warning: 'operator==' must resolve to a function declared within the '__llvm_libc' namespace [llvmlibc-callee-namespace] - REQUIRE(expected == actual); - ^ -/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/test/catch.hpp:2342:14: note: resolves to this declaration - auto operator == ( RhsT const& rhs ) -> BinaryExpr const { - ^ - - -real 0m8.602s -user 0m8.542s -sys 0m0.061s - -=============================================================================== - diff --git a/cpp/lasagna/run-tests-cpp.txt b/cpp/lasagna/run-tests-cpp.txt new file mode 100644 index 00000000..e2a69868 --- /dev/null +++ b/cpp/lasagna/run-tests-cpp.txt @@ -0,0 +1,184 @@ +Running automated test file(s): + + +=============================================================================== + +Running: make clean +rm -rf ./build + +real 0m0.003s +user 0m0.001s +sys 0m0.002s + +=============================================================================== + +export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" + +=============================================================================== + +Running: make coverage | ansifilter +rm -rf ./build +mkdir -pv ./build +mkdir: created directory './build' + +cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi +-- The CXX compiler identification is GNU 10.2.1 +-- Detecting CXX compiler ABI info +-- Detecting CXX compiler ABI info - done +-- Check for working CXX compiler: /usr/bin/c++ - skipped +-- Detecting CXX compile features +-- Detecting CXX compile features - done +-- Configuring done +-- Generating done +-- Build files have been written to: /home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build +make[1]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[2]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +[ 33%] Building CXX object CMakeFiles/lasagna.dir/lasagna_test.cpp.o +[ 66%] Building CXX object CMakeFiles/lasagna.dir/test/tests-main.cpp.o +[100%] Linking CXX executable lasagna +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +[100%] Built target lasagna +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +=============================================================================== +All tests passed (7 assertions in 7 test cases) + +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +[100%] Built target test_lasagna +make[2]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' +make[1]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/build' + +=== All Tests Passed === + +find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$" -print -delete +./build/CMakeFiles/3.22.2/CompilerIdCXX/CMakeCXXCompilerId.gcno +./build/CMakeFiles/lasagna.dir/test/tests-main.cpp.gcno +./build/CMakeFiles/lasagna.dir/test/tests-main.cpp.gcda + +find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$" +./build/CMakeFiles/lasagna.dir/lasagna_test.cpp.gcno +./build/CMakeFiles/lasagna.dir/lasagna_test.cpp.gcda + +gcovr --print-summary +------------------------------------------------------------------------------ + GCC Code Coverage Report +Directory: . +------------------------------------------------------------------------------ +File Lines Exec Cover Missing +------------------------------------------------------------------------------ +lasagna.cpp 7 7 100% +lasagna_test.cpp 49 49 100% +test/catch.hpp 32 27 84% 1642-1643,2227,2229,2560 +------------------------------------------------------------------------------ +TOTAL 88 83 94% +------------------------------------------------------------------------------ +lines: 94.3% (83 out of 88) +branches: 34.6% (36 out of 104) + +real 0m9.390s +user 0m8.629s +sys 0m0.750s + +=============================================================================== + +cmake-format --in-place CMakeLists.txt + +real 0m0.175s +user 0m0.130s +sys 0m0.047s + +=============================================================================== + +cmake-lint CMakeLists.txt +CMakeLists.txt +============== +CMakeLists.txt:15,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` +CMakeLists.txt:17,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` +CMakeLists.txt:52,71: [C0307] Bad indentation: + ) + ^----BodyNode: 1:0->IfBlockNode: 46:0->BodyNode: 46:52->StatementNode: 47:2->TreeNode: 52:71 + +CMakeLists.txt:63,70: [C0307] Bad indentation: + ) + ^----BodyNode: 1:0->IfBlockNode: 61:0->BodyNode: 61:11->StatementNode: 62:2->TreeNode: 63:70 + +CMakeLists.txt:68,02: [C0113] Missing COMMENT in statement which allows it + +Summary +======= +files scanned: 1 +found lint: + Convention: 5 + + +real 0m0.153s +user 0m0.107s +sys 0m0.046s + +=============================================================================== + +Running: clang-format-16 -style=file -i ./lasagna.cpp ./lasagna_test.cpp + +real 0m0.022s +user 0m0.011s +sys 0m0.011s + +=============================================================================== + +Running: ../../.github/citools/cpp/clang-check +clang-check-16 --analyze ./lasagna.cpp ./lasagna_test.cpp + +real 0m0.391s +user 0m0.361s +sys 0m0.030s + + +real 0m0.393s +user 0m0.363s +sys 0m0.030s + +=============================================================================== + +Running: ../../.github/citools/cpp/clang-tidy | head -n 100 +3 warnings generated. +35392 warnings generated. +Suppressed 35391 warnings (35386 in non-user code, 5 NOLINT). +Use -header-filter=.* to display errors from all non-system headers. Use -system-headers to display errors from system headers as well. + +real 0m5.330s +user 0m5.281s +sys 0m0.049s +Running: clang-tidy-16 ./lasagna.cpp ./lasagna_test.cpp +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:1:1: warning: replace macro with enum [modernize-macro-to-enum] +#define OVEN_TIME (40) +^~~~~~~~ + = , +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:1:9: warning: macro 'OVEN_TIME' defines an integral constant; prefer an enum instead [modernize-macro-to-enum] +#define OVEN_TIME (40) + ^ +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna.cpp:2:9: warning: macro 'TIME_PER_LAYER' defines an integral constant; prefer an enum instead [modernize-macro-to-enum] +#define TIME_PER_LAYER (2) + ^ +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:1:11: warning: suspicious #include of file with '.cpp' extension [bugprone-suspicious-include] +#include "lasagna.cpp" + ^ +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:11:5: warning: variable 'actual' of type 'int' can be declared 'const' [misc-const-correctness] + int actual = 40; + ^ + const +/home/vpayno/git_vpayno/exercism-workspace/cpp/lasagna/lasagna_test.cpp:12:5: warning: variable 'expected' of type 'int' can be declared 'const' [misc-const-correctness] + int expected = ovenTime(); + ^ + const + + +real 0m5.332s +user 0m5.281s +sys 0m0.051s + +=============================================================================== + diff --git a/cpp/last-will/.exercism/config.json b/cpp/last-will/.exercism/config.json new file mode 100644 index 00000000..b5d967d6 --- /dev/null +++ b/cpp/last-will/.exercism/config.json @@ -0,0 +1,18 @@ +{ + "authors": [ + "vaeng" + ], + "files": { + "solution": [ + "last_will.cpp" + ], + "test": [ + "last_will_test.cpp" + ], + "exemplar": [ + ".meta/exemplar.cpp" + ] + }, + "icon": "lasagna", + "blurb": "Learn about namespaces by executing an estate." +} diff --git a/cpp/last-will/.exercism/metadata.json b/cpp/last-will/.exercism/metadata.json new file mode 100644 index 00000000..1380e828 --- /dev/null +++ b/cpp/last-will/.exercism/metadata.json @@ -0,0 +1 @@ +{"track":"cpp","exercise":"last-will","id":"355525d5abda443f80cd961ca7912986","url":"https://exercism.org/tracks/cpp/exercises/last-will","handle":"vpayno","is_requester":true,"auto_approve":false} \ No newline at end of file diff --git a/cpp/last-will/CMakeLists.txt b/cpp/last-will/CMakeLists.txt new file mode 100644 index 00000000..aac52b34 --- /dev/null +++ b/cpp/last-will/CMakeLists.txt @@ -0,0 +1,70 @@ +# Basic CMake project +cmake_minimum_required(VERSION 3.5.1) + +# Get the exercise name from the current directory +get_filename_component(exercise ${CMAKE_CURRENT_SOURCE_DIR} NAME) + +# Name the project after the exercise +project(${exercise} CXX) + +# Get a source filename from the exercise name by replacing -'s with _'s +string(REPLACE "-" "_" file ${exercise}) + +# Implementation could be only a header +if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/${file}.cpp) + set(exercise_cpp ${file}.cpp) +else() + set(exercise_cpp "") +endif() + +# Use the common Catch library? +if(EXERCISM_COMMON_CATCH) + # For Exercism track development only + add_executable(${exercise} ${file}_test.cpp $) +elseif(EXERCISM_TEST_SUITE) + # The Exercism test suite is being run, the Docker image already includes a + # pre-built version of Catch. + find_package(Catch2 REQUIRED) + add_executable(${exercise} ${file}_test.cpp) + target_link_libraries(${exercise} PRIVATE Catch2::Catch2WithMain) + # When Catch is installed system wide we need to include a different header, + # we need this define to use the correct one. + target_compile_definitions(${exercise} PRIVATE EXERCISM_TEST_SUITE) +else() + # Build executable from sources and headers + add_executable(${exercise} ${file}_test.cpp test/tests-main.cpp) +endif() + +set_target_properties( + ${exercise} + PROPERTIES CXX_STANDARD 17 + CXX_STANDARD_REQUIRED OFF + CXX_EXTENSIONS OFF) + +set(CMAKE_BUILD_TYPE Debug) + +if("${CMAKE_CXX_COMPILER_ID}" MATCHES "(GNU|Clang)") + set_target_properties( + ${exercise} + PROPERTIES # added "-Wno-unused-parameter" to remove compiler warnings + # should make it easier for students to run their first real code + COMPILE_FLAGS + "-Wall -Wextra -Wpedantic -Werror -Wno-unused-parameter") +endif() + +# Configure to run all the tests? +if(${EXERCISM_RUN_ALL_TESTS}) + target_compile_definitions(${exercise} PRIVATE EXERCISM_RUN_ALL_TESTS) +endif() + +# Tell MSVC not to warn us about unchecked iterators in debug builds +if(${MSVC}) + set_target_properties(${exercise} PROPERTIES COMPILE_DEFINITIONS_DEBUG + _SCL_SECURE_NO_WARNINGS) +endif() + +# Run the tests on every build +add_custom_target( + test_${exercise} ALL + DEPENDS ${exercise} + COMMAND ${exercise}) diff --git a/cpp/last-will/HELP.md b/cpp/last-will/HELP.md new file mode 100644 index 00000000..e6862728 --- /dev/null +++ b/cpp/last-will/HELP.md @@ -0,0 +1,55 @@ +# Help + +## Running the tests + +Running the tests involves running `cmake -G` and then using the build command appropriate for your platform. +Detailed instructions on how to do this can be found on the [Running the Tests](https://exercism.io/tracks/cpp/tests) page for C++ on exercism.io. + +## Passing the Tests + +Get the first test compiling, linking and passing by following the [three +rules of test-driven development](http://butunclebob.com/ArticleS.UncleBob.TheThreeRulesOfTdd). +Create just enough structure by declaring namespaces, functions, classes, +etc., to satisfy any compiler errors and get the test to fail. Then write +just enough code to get the test to pass. Once you've done that, +uncomment the next test by moving the following line past the next test. + +```C++ +#if defined(EXERCISM_RUN_ALL_TESTS) +``` + +This may result in compile errors as new constructs may be invoked that +you haven't yet declared or defined. Again, fix the compile errors minimally +to get a failing test, then change the code minimally to pass the test, +refactor your implementation for readability and expressiveness and then +go on to the next test. + +Try to use standard C++11 facilities in preference to writing your own +low-level algorithms or facilities by hand. + +## Submitting your solution + +You can submit your solution using the `exercism submit last_will.cpp` command. +This command will upload your solution to the Exercism website and print the solution page's URL. + +It's possible to submit an incomplete solution which allows you to: + +- See how others have completed the exercise +- Request help from a mentor + +## Need to get help? + +If you'd like help solving the exercise, check the following pages: + +- The [C++ track's documentation](https://exercism.org/docs/tracks/cpp) +- [Exercism's programming category on the forum](https://forum.exercism.org/c/programming/5) +- The [Frequently Asked Questions](https://exercism.org/docs/using/faqs) + +Should those resources not suffice, you could submit your (incomplete) solution to request mentoring. + +To get help if you're having trouble, you can use one of the following resources: + +- [`c++-faq` tag on StackOverflow](https://stackoverflow.com/tags/c%2b%2b-faq/info) +- [C++ FAQ from isocpp.com](https://isocpp.org/faq) +- [CppReference](http://en.cppreference.com/) is a wiki reference to the C++ language and standard library +- [C traps and pitfalls](http://www.slideshare.net/LegalizeAdulthood/c-traps-and-pitfalls-for-c-programmers) is useful if you are new to C++, but have programmed in C \ No newline at end of file diff --git a/cpp/last-will/HINTS.md b/cpp/last-will/HINTS.md new file mode 100644 index 00000000..61bf1f94 --- /dev/null +++ b/cpp/last-will/HINTS.md @@ -0,0 +1,20 @@ +# Hints + +## General + +- Do not change the code in the families' namespaces. +- Do not copy the values into your code, call the functions. + +## 1. Take your seat in front of the families and lay out your papers + +- The namespace has to be called `estate_executor` for the tests to work. + +## 2. Find the secret account number + +Each `bank_number_part` has to be called with the `secret_modifier` from the parameter list. + +## 3. Last step: Enter the secret code + +- You can call functions from nested namespaces like this `outer_namespace::inner_namespac::my_function`. + +- Take care to add the blue and the red fragments separately before multiplicating both parts. \ No newline at end of file diff --git a/cpp/last-will/Makefile b/cpp/last-will/Makefile new file mode 100644 index 00000000..62b4b657 --- /dev/null +++ b/cpp/last-will/Makefile @@ -0,0 +1,23 @@ +.PHONY: clean +clean: + rm -rf ./build + +.PHONY: test +test: clean + mkdir -pv ./build + + @printf "\n" + @# each line is executed in a subshell, we have to daisy chain them so they + @# run in the build directory + cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi + +.PHONY: coverage +coverage: test + @printf "\n" + find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$$" -print -delete + + @printf "\n" + find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$$" + + @printf "\n" + gcovr --print-summary diff --git a/cpp/last-will/README.md b/cpp/last-will/README.md new file mode 100644 index 00000000..02119e8a --- /dev/null +++ b/cpp/last-will/README.md @@ -0,0 +1,88 @@ +# Last Will + +Welcome to Last Will on Exercism's C++ Track. +If you need help running the tests or submitting your code, check out `HELP.md`. +If you get stuck on the exercise, check out `HINTS.md`, but try and solve it without using those first :) + +## Introduction + +## Namespaces + +An important method for code organization are namespaces. +Two functions might have a naming collision, which can be resolved by putting them in different namespaces. +Namespaces can be nested, which might help to structure big code bases. +Access to the namespaces is done via the scope-resolution operator `::`. + +The example below shows the use of two different `foo` functions. +They are used together by prefixing their respective namespaces. + +```cpp +namespace my_ns { + int foo() { + return 44; + } + namespace my_inner_ns { + int baz() { + return 90; + } + } +} +namespace my_other_ns { + int foo() { + return -2; + } +} + +int myresult{my_ns::foo() + my_other_ns::foo() * my_ns::my_inner_ns::baz()}; +``` + +## Instructions + +You work for a prestigious law firm that is specialized in handling unique testament requests. + +In this exercise, you are going to open a mysterious vault. +You are the executor of the estate and will assemble the long-kept secret codes of three families to get an account number and the matching code. + +To prevent any family from opening the vault alone, it can only be opened by combining their knowledge with a secret modifier that you know from the last will. + +You have three tasks, all related to helping the families to open the vault. + +## 1. Take your seat in front of the families and lay out your papers + +Define a namespace called `estate_executor`. +The code from the next tasks should be defined in the body of the `estate_executor` namespace. + +```cpp +namespace some_name { + // The space between the curly brackets + // is called body of the namespace. +} +``` + +## 2. Find the secret account number + +This is your big moment. +Only you have the secret modifier key to reveal the secret account number. + +Define the `assemble_account_number(int secret_modifier)` function that takes the secret modifier as an argument and returns the assembled account number as an `int`. + +To get the correct number, you have to sum up the `bank_number_part` from each of the three families. + +## 3. Last step: Enter the secret code + +The instructions in the testament ask you to add all the blue and then all the red fragments. +The resulting code is obtained by multiplying both sums. + +Define the `assemble_code()` function that returns the resulting code by combining the fragments from the three families to a single `integer`. +The function does not have any arguments and relies solely on the information in the relevant namespaces from the families. + +## Source + +### Created by + +- @vaeng + +### My Solution + +- [my solution]() +- [run-tests](./run-tests-cpp.txt) diff --git a/cpp/last-will/compile_commands.json b/cpp/last-will/compile_commands.json new file mode 100644 index 00000000..1123820d --- /dev/null +++ b/cpp/last-will/compile_commands.json @@ -0,0 +1,12 @@ +[ + { + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -Wno-unused-parameter -std=c++17 -o CMakeFiles/last-will.dir/last_will_test.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will_test.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will_test.cpp" + }, + { + "directory": "/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build", + "command": "/usr/bin/c++ -DEXERCISM_RUN_ALL_TESTS --coverage -g -Wall -Wextra -Wpedantic -Werror -Wno-unused-parameter -std=c++17 -o CMakeFiles/last-will.dir/test/tests-main.cpp.o -c /home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/test/tests-main.cpp", + "file": "/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/test/tests-main.cpp" + } +] diff --git a/cpp/last-will/compile_flags.txt b/cpp/last-will/compile_flags.txt new file mode 100644 index 00000000..5dfc4cfe --- /dev/null +++ b/cpp/last-will/compile_flags.txt @@ -0,0 +1,8 @@ +-std=c++17 +-stdlib=libstdc++ +-g +-Wall +-Wextra +-Wpedantic +-Werror +-Wno-unused-parameter diff --git a/cpp/last-will/last_will.cpp b/cpp/last-will/last_will.cpp new file mode 100644 index 00000000..6d765eab --- /dev/null +++ b/cpp/last-will/last_will.cpp @@ -0,0 +1,71 @@ +// Enter your code below the lines of the families' information + +// Secret knowledge of the Zhang family: +namespace zhang { +int bank_number_part(int secret_modifier) { + int zhang_part{8'541}; + return (zhang_part * secret_modifier) % 10000; +} +namespace red { +int code_fragment() { return 512; } +} // namespace red +namespace blue { +int code_fragment() { return 677; } +} // namespace blue +} // namespace zhang + +// Secret knowledge of the Khan family: +namespace khan { +int bank_number_part(int secret_modifier) { + int khan_part{4'142}; + return (khan_part * secret_modifier) % 10000; +} +namespace red { +int code_fragment() { return 148; } +} // namespace red +namespace blue { +int code_fragment() { return 875; } +} // namespace blue +} // namespace khan + +// Secret knowledge of the Garcia family: +namespace garcia { +int bank_number_part(int secret_modifier) { + int garcia_part{4'023}; + return (garcia_part * secret_modifier) % 10000; +} +namespace red { +int code_fragment() { return 118; } +} // namespace red +namespace blue { +int code_fragment() { return 923; } +} // namespace blue +} // namespace garcia + +// Enter your code below +namespace estate_executor { +int assemble_account_number(int secret_modifier) { + const int part1{zhang::bank_number_part(secret_modifier)}; + const int part2{khan::bank_number_part(secret_modifier)}; + const int part3{garcia::bank_number_part(secret_modifier)}; + + return part1 + part2 + part3; +} + +int assemble_code() { + int red{0}; + int blue{0}; + int code{0}; + + red += zhang::red::code_fragment(); + blue += zhang::blue::code_fragment(); + red += khan::red::code_fragment(); + blue += khan::blue::code_fragment(); + red += garcia::red::code_fragment(); + blue += garcia::blue::code_fragment(); + + code = red * blue; + + return code; +} +} // namespace estate_executor diff --git a/cpp/last-will/last_will.plist b/cpp/last-will/last_will.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/last-will/last_will.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/last-will/last_will_test.cpp b/cpp/last-will/last_will_test.cpp new file mode 100644 index 00000000..084ed664 --- /dev/null +++ b/cpp/last-will/last_will_test.cpp @@ -0,0 +1,61 @@ +// Trick to let the code compile, even if the function has not been implemented: +namespace estate_executor { +int assemble_account_number(int) __attribute__((weak)); +int assemble_code() __attribute__((weak)); +} // namespace estate_executor + +#include "last_will.cpp" +#ifdef EXERCISM_TEST_SUITE +#include +#else +#include "test/catch.hpp" +#endif + +using namespace std; + +TEST_CASE("Family secrets have not been altered") { + zhang::bank_number_part(1); + REQUIRE(zhang::bank_number_part(1) == 8541); + REQUIRE(zhang::bank_number_part(3) == 8541 * 3 % 10'000); + REQUIRE(khan::bank_number_part(1) == 4142); + REQUIRE(khan::bank_number_part(3) == 4142 * 3 % 10'000); + REQUIRE(garcia::bank_number_part(1) == 4023); + REQUIRE(garcia::bank_number_part(3) == 4023 * 3 % 10'000); + + REQUIRE(zhang::red::code_fragment() == 512); + REQUIRE(khan::red::code_fragment() == 148); + REQUIRE(garcia::red::code_fragment() == 118); + + REQUIRE(zhang::blue::code_fragment() == 677); + REQUIRE(khan::blue::code_fragment() == 875); + REQUIRE(garcia::blue::code_fragment() == 923); +} + +TEST_CASE("Account number assembly function exists in correct namespace") { + REQUIRE_NOTHROW(estate_executor::assemble_account_number(0)); +} + +#if defined(EXERCISM_RUN_ALL_TESTS) + +TEST_CASE("Account number assembly works correctly") { + int account_with_secret_1{16706}; + int account_with_secret_23{14238}; + + REQUIRE(estate_executor::assemble_account_number(1) == + account_with_secret_1); + REQUIRE(estate_executor::assemble_account_number(23) == + account_with_secret_23); +} + +TEST_CASE( + "Code fragment number assembly function exists in correct namespace") { + REQUIRE_NOTHROW(estate_executor::assemble_code()); +} + +TEST_CASE("Code fragments fit correctly") { + int final_code{1925550}; + + REQUIRE(estate_executor::assemble_code() == final_code); +} + +#endif diff --git a/cpp/last-will/last_will_test.plist b/cpp/last-will/last_will_test.plist new file mode 100644 index 00000000..4fa1e477 --- /dev/null +++ b/cpp/last-will/last_will_test.plist @@ -0,0 +1,14 @@ + + + + + clang_version +Debian clang version 16.0.6 (++20230610113348+7cbf1a259152-1~exp1~20230610233446.99) + diagnostics + + + files + + + + diff --git a/cpp/last-will/run-tests-cpp.txt b/cpp/last-will/run-tests-cpp.txt new file mode 100644 index 00000000..f3c78d3f --- /dev/null +++ b/cpp/last-will/run-tests-cpp.txt @@ -0,0 +1,182 @@ +Running automated test file(s): + + +=============================================================================== + +Running: make clean +rm -rf ./build + +real 0m0.011s +user 0m0.002s +sys 0m0.009s + +=============================================================================== + +export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage" + +=============================================================================== + +Running: make coverage | ansifilter +rm -rf ./build +mkdir -pv ./build +mkdir: created directory './build' + +cd ./build; export LDFLAGS="-lgcov --coverage" CXXFLAGS="--coverage"; cmake -DEXERCISM_RUN_ALL_TESTS=1 -DCMAKE_EXPORT_COMPILE_COMMANDS=1 -G 'Unix Makefiles' ../; if make; then printf "\n=== All Tests Passed ===\n"; else printf "\n=== Test Failure ===\n"; false; fi +-- The CXX compiler identification is GNU 10.2.1 +-- Detecting CXX compiler ABI info +-- Detecting CXX compiler ABI info - done +-- Check for working CXX compiler: /usr/bin/c++ - skipped +-- Detecting CXX compile features +-- Detecting CXX compile features - done +-- Configuring done +-- Generating done +-- Build files have been written to: /home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build +make[1]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[2]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +[ 33%] Building CXX object CMakeFiles/last-will.dir/last_will_test.cpp.o +[ 66%] Building CXX object CMakeFiles/last-will.dir/test/tests-main.cpp.o +[100%] Linking CXX executable last-will +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +[100%] Built target last-will +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[3]: Entering directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +=============================================================================== +All tests passed (17 assertions in 5 test cases) + +make[3]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +[100%] Built target test_last-will +make[2]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' +make[1]: Leaving directory '/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/build' + +=== All Tests Passed === + +find . -regextype posix-egrep -regex "^.*(tests-main|CompilerId).*[.](gcda|gcno)$" -print -delete +./build/CMakeFiles/3.22.2/CompilerIdCXX/CMakeCXXCompilerId.gcno +./build/CMakeFiles/last-will.dir/test/tests-main.cpp.gcno +./build/CMakeFiles/last-will.dir/test/tests-main.cpp.gcda + +find . -regextype posix-egrep -regex "^.*[.](gcda|gcno)$" +./build/CMakeFiles/last-will.dir/last_will_test.cpp.gcno +./build/CMakeFiles/last-will.dir/last_will_test.cpp.gcda + +gcovr --print-summary +------------------------------------------------------------------------------ + GCC Code Coverage Report +Directory: . +------------------------------------------------------------------------------ +File Lines Exec Cover Missing +------------------------------------------------------------------------------ +last_will.cpp 32 32 100% +last_will_test.cpp 31 31 100% +test/catch.hpp 32 27 84% 1642-1643,2227,2229,2560 +------------------------------------------------------------------------------ +TOTAL 95 90 94% +------------------------------------------------------------------------------ +lines: 94.7% (90 out of 95) +branches: 35.4% (87 out of 246) + +real 0m8.632s +user 0m7.922s +sys 0m0.706s + +=============================================================================== + +cmake-format --in-place CMakeLists.txt + +real 0m0.304s +user 0m0.193s +sys 0m0.085s + +=============================================================================== + +cmake-lint CMakeLists.txt +CMakeLists.txt +============== +CMakeLists.txt:15,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` +CMakeLists.txt:17,06: [C0103] Invalid directory variable name "exercise_cpp" doesn't match `[A-Z][0-9A-Z_]+|_[0-9a-z_]+` +CMakeLists.txt:52,71: [C0307] Bad indentation: + ) + ^----BodyNode: 1:0->IfBlockNode: 46:0->BodyNode: 46:52->StatementNode: 47:2->TreeNode: 52:71 + +CMakeLists.txt:63,70: [C0307] Bad indentation: + ) + ^----BodyNode: 1:0->IfBlockNode: 61:0->BodyNode: 61:11->StatementNode: 62:2->TreeNode: 63:70 + +CMakeLists.txt:68,02: [C0113] Missing COMMENT in statement which allows it + +Summary +======= +files scanned: 1 +found lint: + Convention: 5 + + +real 0m0.156s +user 0m0.105s +sys 0m0.051s + +=============================================================================== + +Running: clang-format-16 -style=file -i ./last_will.cpp ./last_will_test.cpp + +real 0m0.021s +user 0m0.009s +sys 0m0.011s + +=============================================================================== + +Running: ../../.github/citools/cpp/clang-check +clang-check-16 --analyze ./last_will.cpp ./last_will_test.cpp + +real 0m0.398s +user 0m0.358s +sys 0m0.039s + + +real 0m0.399s +user 0m0.358s +sys 0m0.041s + +=============================================================================== + +Running: ../../.github/citools/cpp/clang-tidy | head -n 100 +3 warnings generated. +35406 warnings generated. +Suppressed 35407 warnings (35401 in non-user code, 6 NOLINT). +Use -header-filter=.* to display errors from all non-system headers. Use -system-headers to display errors from system headers as well. + +real 0m5.123s +user 0m5.057s +sys 0m0.065s +Running: clang-tidy-16 ./last_will.cpp ./last_will_test.cpp +/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will.cpp:6:5: warning: variable 'zhang_part' of type 'int' can be declared 'const' [misc-const-correctness] + int zhang_part{8'541}; + ^ + const +/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will.cpp:20:5: warning: variable 'khan_part' of type 'int' can be declared 'const' [misc-const-correctness] + int khan_part{4'142}; + ^ + const +/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will.cpp:34:5: warning: variable 'garcia_part' of type 'int' can be declared 'const' [misc-const-correctness] + int garcia_part{4'023}; + ^ + const +/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will_test.cpp:3:32: warning: all parameters should be named in a function [readability-named-parameter] +int assemble_account_number(int) __attribute__((weak)); + ^ + /*secret_modifier*/ +/home/vpayno/git_vpayno/exercism-workspace/cpp/last-will/last_will_test.cpp:7:11: warning: suspicious #include of file with '.cpp' extension [bugprone-suspicious-include] +#include "last_will.cpp" + ^ + + +real 0m5.125s +user 0m5.059s +sys 0m0.067s + +=============================================================================== + diff --git a/cpp/last-will/test/catch.hpp b/cpp/last-will/test/catch.hpp new file mode 100644 index 00000000..36eaeb27 --- /dev/null +++ b/cpp/last-will/test/catch.hpp @@ -0,0 +1,17937 @@ +/* + * Catch v2.13.6 + * Generated: 2021-04-16 18:23:38.044268 + * ---------------------------------------------------------- + * This file has been merged from multiple headers. Please don't edit it directly + * Copyright (c) 2021 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 +// start catch.hpp + + +#define CATCH_VERSION_MAJOR 2 +#define CATCH_VERSION_MINOR 13 +#define CATCH_VERSION_PATCH 6 + +#ifdef __clang__ +# pragma clang system_header +#elif defined __GNUC__ +# pragma GCC system_header +#endif + +// start 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 push +# pragma clang diagnostic ignored "-Wpadded" +# pragma clang diagnostic ignored "-Wswitch-enum" +# pragma clang diagnostic ignored "-Wcovered-switch-default" +# endif +#elif defined __GNUC__ + // Because REQUIREs trigger GCC's -Wparentheses, and because still + // supported version of g++ have only buggy support for _Pragmas, + // Wparentheses have to be suppressed globally. +# pragma GCC diagnostic ignored "-Wparentheses" // See #674 for details + +# pragma GCC diagnostic push +# pragma GCC diagnostic ignored "-Wunused-variable" +# pragma GCC diagnostic ignored "-Wpadded" +#endif +// end catch_suppress_warnings.h +#if defined(CATCH_CONFIG_MAIN) || defined(CATCH_CONFIG_RUNNER) +# define CATCH_IMPL +# define CATCH_CONFIG_ALL_PARTS +#endif + +// In the impl file, we want to have access to all parts of the headers +// Can also be used to sanely support PCHs +#if defined(CATCH_CONFIG_ALL_PARTS) +# define CATCH_CONFIG_EXTERNAL_INTERFACES +# if defined(CATCH_CONFIG_DISABLE_MATCHERS) +# undef CATCH_CONFIG_DISABLE_MATCHERS +# endif +# if !defined(CATCH_CONFIG_ENABLE_CHRONO_STRINGMAKER) +# define CATCH_CONFIG_ENABLE_CHRONO_STRINGMAKER +# endif +#endif + +#if !defined(CATCH_CONFIG_IMPL_ONLY) +// start catch_platform.h + +// See e.g.: +// https://opensource.apple.com/source/CarbonHeaders/CarbonHeaders-18.1/TargetConditionals.h.auto.html +#ifdef __APPLE__ +# include +# if (defined(TARGET_OS_OSX) && TARGET_OS_OSX == 1) || \ + (defined(TARGET_OS_MAC) && TARGET_OS_MAC == 1) +# define CATCH_PLATFORM_MAC +# elif (defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE == 1) +# define CATCH_PLATFORM_IPHONE +# endif + +#elif defined(linux) || defined(__linux) || defined(__linux__) +# define CATCH_PLATFORM_LINUX + +#elif defined(WIN32) || defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER) || defined(__MINGW32__) +# define CATCH_PLATFORM_WINDOWS +#endif + +// end catch_platform.h + +#ifdef CATCH_IMPL +# ifndef CLARA_CONFIG_MAIN +# define CLARA_CONFIG_MAIN_NOT_DEFINED +# define CLARA_CONFIG_MAIN +# endif +#endif + +// start catch_user_interfaces.h + +namespace Catch { + unsigned int rngSeed(); +} + +// end catch_user_interfaces.h +// start catch_tag_alias_autoregistrar.h + +// start catch_common.h + +// start catch_compiler_capabilities.h + +// Detect a number of compiler features - by compiler +// The following features are defined: +// +// CATCH_CONFIG_COUNTER : is the __COUNTER__ macro supported? +// CATCH_CONFIG_WINDOWS_SEH : is Windows SEH supported? +// CATCH_CONFIG_POSIX_SIGNALS : are POSIX signals supported? +// CATCH_CONFIG_DISABLE_EXCEPTIONS : Are exceptions enabled? +// **************** +// 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_NO_POSIX_SIGNALS) 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. + +#ifdef __cplusplus + +# if (__cplusplus >= 201402L) || (defined(_MSVC_LANG) && _MSVC_LANG >= 201402L) +# define CATCH_CPP14_OR_GREATER +# endif + +# if (__cplusplus >= 201703L) || (defined(_MSVC_LANG) && _MSVC_LANG >= 201703L) +# define CATCH_CPP17_OR_GREATER +# endif + +#endif + +// Only GCC compiler should be used in this block, so other compilers trying to +// mask themselves as GCC should be ignored. +#if defined(__GNUC__) && !defined(__clang__) && !defined(__ICC) && !defined(__CUDACC__) && !defined(__LCC__) +# define CATCH_INTERNAL_START_WARNINGS_SUPPRESSION _Pragma( "GCC diagnostic push" ) +# define CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION _Pragma( "GCC diagnostic pop" ) + +# define CATCH_INTERNAL_IGNORE_BUT_WARN(...) (void)__builtin_constant_p(__VA_ARGS__) + +#endif + +#if defined(__clang__) + +# define CATCH_INTERNAL_START_WARNINGS_SUPPRESSION _Pragma( "clang diagnostic push" ) +# define CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION _Pragma( "clang diagnostic pop" ) + +// As of this writing, IBM XL's implementation of __builtin_constant_p has a bug +// which results in calls to destructors being emitted for each temporary, +// without a matching initialization. In practice, this can result in something +// like `std::string::~string` being called on an uninitialized value. +// +// For example, this code will likely segfault under IBM XL: +// ``` +// REQUIRE(std::string("12") + "34" == "1234") +// ``` +// +// Therefore, `CATCH_INTERNAL_IGNORE_BUT_WARN` is not implemented. +# if !defined(__ibmxl__) && !defined(__CUDACC__) +# define CATCH_INTERNAL_IGNORE_BUT_WARN(...) (void)__builtin_constant_p(__VA_ARGS__) /* NOLINT(cppcoreguidelines-pro-type-vararg, hicpp-vararg) */ +# endif + +# define CATCH_INTERNAL_SUPPRESS_GLOBALS_WARNINGS \ + _Pragma( "clang diagnostic ignored \"-Wexit-time-destructors\"" ) \ + _Pragma( "clang diagnostic ignored \"-Wglobal-constructors\"") + +# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS \ + _Pragma( "clang diagnostic ignored \"-Wparentheses\"" ) + +# define CATCH_INTERNAL_SUPPRESS_UNUSED_WARNINGS \ + _Pragma( "clang diagnostic ignored \"-Wunused-variable\"" ) + +# define CATCH_INTERNAL_SUPPRESS_ZERO_VARIADIC_WARNINGS \ + _Pragma( "clang diagnostic ignored \"-Wgnu-zero-variadic-macro-arguments\"" ) + +# define CATCH_INTERNAL_SUPPRESS_UNUSED_TEMPLATE_WARNINGS \ + _Pragma( "clang diagnostic ignored \"-Wunused-template\"" ) + +#endif // __clang__ + +//////////////////////////////////////////////////////////////////////////////// +// Assume that non-Windows platforms support posix signals by default +#if !defined(CATCH_PLATFORM_WINDOWS) + #define CATCH_INTERNAL_CONFIG_POSIX_SIGNALS +#endif + +//////////////////////////////////////////////////////////////////////////////// +// We know some environments not to support full POSIX signals +#if defined(__CYGWIN__) || defined(__QNX__) || defined(__EMSCRIPTEN__) || defined(__DJGPP__) + #define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS +#endif + +#ifdef __OS400__ +# define CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS +# define CATCH_CONFIG_COLOUR_NONE +#endif + +//////////////////////////////////////////////////////////////////////////////// +// Android somehow still does not support std::to_string +#if defined(__ANDROID__) +# define CATCH_INTERNAL_CONFIG_NO_CPP11_TO_STRING +# define CATCH_INTERNAL_CONFIG_ANDROID_LOGWRITE +#endif + +//////////////////////////////////////////////////////////////////////////////// +// Not all Windows environments support SEH properly +#if defined(__MINGW32__) +# define CATCH_INTERNAL_CONFIG_NO_WINDOWS_SEH +#endif + +//////////////////////////////////////////////////////////////////////////////// +// PS4 +#if defined(__ORBIS__) +# define CATCH_INTERNAL_CONFIG_NO_NEW_CAPTURE +#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 +// some versions of cygwin (most) do not support std::to_string. Use the libstd check. +// https://gcc.gnu.org/onlinedocs/gcc-4.8.2/libstdc++/api/a01053_source.html line 2812-2813 +# if !((__cplusplus >= 201103L) && defined(_GLIBCXX_USE_C99) \ + && !defined(_GLIBCXX_HAVE_BROKEN_VSWPRINTF)) + +# define CATCH_INTERNAL_CONFIG_NO_CPP11_TO_STRING + +# endif +#endif // __CYGWIN__ + +//////////////////////////////////////////////////////////////////////////////// +// Visual C++ +#if defined(_MSC_VER) + +# define CATCH_INTERNAL_START_WARNINGS_SUPPRESSION __pragma( warning(push) ) +# define CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION __pragma( warning(pop) ) + +// Universal Windows platform does not support SEH +// Or console colours (or console at all...) +# if defined(WINAPI_FAMILY) && (WINAPI_FAMILY == WINAPI_FAMILY_APP) +# define CATCH_CONFIG_COLOUR_NONE +# else +# define CATCH_INTERNAL_CONFIG_WINDOWS_SEH +# endif + +// MSVC traditional preprocessor needs some workaround for __VA_ARGS__ +// _MSVC_TRADITIONAL == 0 means new conformant preprocessor +// _MSVC_TRADITIONAL == 1 means old traditional non-conformant preprocessor +# if !defined(__clang__) // Handle Clang masquerading for msvc +# if !defined(_MSVC_TRADITIONAL) || (defined(_MSVC_TRADITIONAL) && _MSVC_TRADITIONAL) +# define CATCH_INTERNAL_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR +# endif // MSVC_TRADITIONAL +# endif // __clang__ + +#endif // _MSC_VER + +#if defined(_REENTRANT) || defined(_MSC_VER) +// Enable async processing, as -pthread is specified or no additional linking is required +# define CATCH_INTERNAL_CONFIG_USE_ASYNC +#endif // _MSC_VER + +//////////////////////////////////////////////////////////////////////////////// +// Check if we are compiled with -fno-exceptions or equivalent +#if defined(__EXCEPTIONS) || defined(__cpp_exceptions) || defined(_CPPUNWIND) +# define CATCH_INTERNAL_CONFIG_EXCEPTIONS_ENABLED +#endif + +//////////////////////////////////////////////////////////////////////////////// +// DJGPP +#ifdef __DJGPP__ +# define CATCH_INTERNAL_CONFIG_NO_WCHAR +#endif // __DJGPP__ + +//////////////////////////////////////////////////////////////////////////////// +// Embarcadero C++Build +#if defined(__BORLANDC__) + #define CATCH_INTERNAL_CONFIG_POLYFILL_ISNAN +#endif + +//////////////////////////////////////////////////////////////////////////////// + +// Use of __COUNTER__ is suppressed during code analysis in +// CLion/AppCode 2017.2.x and former, because __COUNTER__ is not properly +// handled by it. +// Otherwise all supported compilers support COUNTER macro, +// but user still might want to turn it off +#if ( !defined(__JETBRAINS_IDE__) || __JETBRAINS_IDE__ >= 20170300L ) + #define CATCH_INTERNAL_CONFIG_COUNTER +#endif + +//////////////////////////////////////////////////////////////////////////////// + +// RTX is a special version of Windows that is real time. +// This means that it is detected as Windows, but does not provide +// the same set of capabilities as real Windows does. +#if defined(UNDER_RTSS) || defined(RTX64_BUILD) + #define CATCH_INTERNAL_CONFIG_NO_WINDOWS_SEH + #define CATCH_INTERNAL_CONFIG_NO_ASYNC + #define CATCH_CONFIG_COLOUR_NONE +#endif + +#if !defined(_GLIBCXX_USE_C99_MATH_TR1) +#define CATCH_INTERNAL_CONFIG_GLOBAL_NEXTAFTER +#endif + +// Various stdlib support checks that require __has_include +#if defined(__has_include) + // Check if string_view is available and usable + #if __has_include() && defined(CATCH_CPP17_OR_GREATER) + # define CATCH_INTERNAL_CONFIG_CPP17_STRING_VIEW + #endif + + // Check if optional is available and usable + # if __has_include() && defined(CATCH_CPP17_OR_GREATER) + # define CATCH_INTERNAL_CONFIG_CPP17_OPTIONAL + # endif // __has_include() && defined(CATCH_CPP17_OR_GREATER) + + // Check if byte is available and usable + # if __has_include() && defined(CATCH_CPP17_OR_GREATER) + # include + # if __cpp_lib_byte > 0 + # define CATCH_INTERNAL_CONFIG_CPP17_BYTE + # endif + # endif // __has_include() && defined(CATCH_CPP17_OR_GREATER) + + // Check if variant is available and usable + # if __has_include() && defined(CATCH_CPP17_OR_GREATER) + # if defined(__clang__) && (__clang_major__ < 8) + // work around clang bug with libstdc++ https://bugs.llvm.org/show_bug.cgi?id=31852 + // fix should be in clang 8, workaround in libstdc++ 8.2 + # include + # if defined(__GLIBCXX__) && defined(_GLIBCXX_RELEASE) && (_GLIBCXX_RELEASE < 9) + # define CATCH_CONFIG_NO_CPP17_VARIANT + # else + # define CATCH_INTERNAL_CONFIG_CPP17_VARIANT + # endif // defined(__GLIBCXX__) && defined(_GLIBCXX_RELEASE) && (_GLIBCXX_RELEASE < 9) + # else + # define CATCH_INTERNAL_CONFIG_CPP17_VARIANT + # endif // defined(__clang__) && (__clang_major__ < 8) + # endif // __has_include() && defined(CATCH_CPP17_OR_GREATER) +#endif // defined(__has_include) + +#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_WINDOWS_SEH) && !defined(CATCH_CONFIG_NO_WINDOWS_SEH) && !defined(CATCH_CONFIG_WINDOWS_SEH) && !defined(CATCH_INTERNAL_CONFIG_NO_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_POSIX_SIGNALS) && !defined(CATCH_INTERNAL_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_NO_POSIX_SIGNALS) && !defined(CATCH_CONFIG_POSIX_SIGNALS) +# define CATCH_CONFIG_POSIX_SIGNALS +#endif +// This is set by default, because we assume that compilers with no wchar_t support are just rare exceptions. +#if !defined(CATCH_INTERNAL_CONFIG_NO_WCHAR) && !defined(CATCH_CONFIG_NO_WCHAR) && !defined(CATCH_CONFIG_WCHAR) +# define CATCH_CONFIG_WCHAR +#endif + +#if !defined(CATCH_INTERNAL_CONFIG_NO_CPP11_TO_STRING) && !defined(CATCH_CONFIG_NO_CPP11_TO_STRING) && !defined(CATCH_CONFIG_CPP11_TO_STRING) +# define CATCH_CONFIG_CPP11_TO_STRING +#endif + +#if defined(CATCH_INTERNAL_CONFIG_CPP17_OPTIONAL) && !defined(CATCH_CONFIG_NO_CPP17_OPTIONAL) && !defined(CATCH_CONFIG_CPP17_OPTIONAL) +# define CATCH_CONFIG_CPP17_OPTIONAL +#endif + +#if defined(CATCH_INTERNAL_CONFIG_CPP17_STRING_VIEW) && !defined(CATCH_CONFIG_NO_CPP17_STRING_VIEW) && !defined(CATCH_CONFIG_CPP17_STRING_VIEW) +# define CATCH_CONFIG_CPP17_STRING_VIEW +#endif + +#if defined(CATCH_INTERNAL_CONFIG_CPP17_VARIANT) && !defined(CATCH_CONFIG_NO_CPP17_VARIANT) && !defined(CATCH_CONFIG_CPP17_VARIANT) +# define CATCH_CONFIG_CPP17_VARIANT +#endif + +#if defined(CATCH_INTERNAL_CONFIG_CPP17_BYTE) && !defined(CATCH_CONFIG_NO_CPP17_BYTE) && !defined(CATCH_CONFIG_CPP17_BYTE) +# define CATCH_CONFIG_CPP17_BYTE +#endif + +#if defined(CATCH_CONFIG_EXPERIMENTAL_REDIRECT) +# define CATCH_INTERNAL_CONFIG_NEW_CAPTURE +#endif + +#if defined(CATCH_INTERNAL_CONFIG_NEW_CAPTURE) && !defined(CATCH_INTERNAL_CONFIG_NO_NEW_CAPTURE) && !defined(CATCH_CONFIG_NO_NEW_CAPTURE) && !defined(CATCH_CONFIG_NEW_CAPTURE) +# define CATCH_CONFIG_NEW_CAPTURE +#endif + +#if !defined(CATCH_INTERNAL_CONFIG_EXCEPTIONS_ENABLED) && !defined(CATCH_CONFIG_DISABLE_EXCEPTIONS) +# define CATCH_CONFIG_DISABLE_EXCEPTIONS +#endif + +#if defined(CATCH_INTERNAL_CONFIG_POLYFILL_ISNAN) && !defined(CATCH_CONFIG_NO_POLYFILL_ISNAN) && !defined(CATCH_CONFIG_POLYFILL_ISNAN) +# define CATCH_CONFIG_POLYFILL_ISNAN +#endif + +#if defined(CATCH_INTERNAL_CONFIG_USE_ASYNC) && !defined(CATCH_INTERNAL_CONFIG_NO_ASYNC) && !defined(CATCH_CONFIG_NO_USE_ASYNC) && !defined(CATCH_CONFIG_USE_ASYNC) +# define CATCH_CONFIG_USE_ASYNC +#endif + +#if defined(CATCH_INTERNAL_CONFIG_ANDROID_LOGWRITE) && !defined(CATCH_CONFIG_NO_ANDROID_LOGWRITE) && !defined(CATCH_CONFIG_ANDROID_LOGWRITE) +# define CATCH_CONFIG_ANDROID_LOGWRITE +#endif + +#if defined(CATCH_INTERNAL_CONFIG_GLOBAL_NEXTAFTER) && !defined(CATCH_CONFIG_NO_GLOBAL_NEXTAFTER) && !defined(CATCH_CONFIG_GLOBAL_NEXTAFTER) +# define CATCH_CONFIG_GLOBAL_NEXTAFTER +#endif + +// Even if we do not think the compiler has that warning, we still have +// to provide a macro that can be used by the code. +#if !defined(CATCH_INTERNAL_START_WARNINGS_SUPPRESSION) +# define CATCH_INTERNAL_START_WARNINGS_SUPPRESSION +#endif +#if !defined(CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION) +# define CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION +#endif +#if !defined(CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_PARENTHESES_WARNINGS +#endif +#if !defined(CATCH_INTERNAL_SUPPRESS_GLOBALS_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_GLOBALS_WARNINGS +#endif +#if !defined(CATCH_INTERNAL_SUPPRESS_UNUSED_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_UNUSED_WARNINGS +#endif +#if !defined(CATCH_INTERNAL_SUPPRESS_ZERO_VARIADIC_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_ZERO_VARIADIC_WARNINGS +#endif + +// The goal of this macro is to avoid evaluation of the arguments, but +// still have the compiler warn on problems inside... +#if !defined(CATCH_INTERNAL_IGNORE_BUT_WARN) +# define CATCH_INTERNAL_IGNORE_BUT_WARN(...) +#endif + +#if defined(__APPLE__) && defined(__apple_build_version__) && (__clang_major__ < 10) +# undef CATCH_INTERNAL_SUPPRESS_UNUSED_TEMPLATE_WARNINGS +#elif defined(__clang__) && (__clang_major__ < 5) +# undef CATCH_INTERNAL_SUPPRESS_UNUSED_TEMPLATE_WARNINGS +#endif + +#if !defined(CATCH_INTERNAL_SUPPRESS_UNUSED_TEMPLATE_WARNINGS) +# define CATCH_INTERNAL_SUPPRESS_UNUSED_TEMPLATE_WARNINGS +#endif + +#if defined(CATCH_CONFIG_DISABLE_EXCEPTIONS) +#define CATCH_TRY if ((true)) +#define CATCH_CATCH_ALL if ((false)) +#define CATCH_CATCH_ANON(type) if ((false)) +#else +#define CATCH_TRY try +#define CATCH_CATCH_ALL catch (...) +#define CATCH_CATCH_ANON(type) catch (type) +#endif + +#if defined(CATCH_INTERNAL_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR) && !defined(CATCH_CONFIG_NO_TRADITIONAL_MSVC_PREPROCESSOR) && !defined(CATCH_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR) +#define CATCH_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR +#endif + +// end catch_compiler_capabilities.h +#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 + +#include +#include +#include + +// We need a dummy global operator<< so we can bring it into Catch namespace later +struct Catch_global_namespace_dummy {}; +std::ostream& operator<<(std::ostream&, Catch_global_namespace_dummy); + +namespace Catch { + + struct CaseSensitive { enum Choice { + Yes, + No + }; }; + + class NonCopyable { + NonCopyable( NonCopyable const& ) = delete; + NonCopyable( NonCopyable && ) = delete; + NonCopyable& operator = ( NonCopyable const& ) = delete; + NonCopyable& operator = ( NonCopyable && ) = delete; + + protected: + NonCopyable(); + virtual ~NonCopyable(); + }; + + struct SourceLineInfo { + + SourceLineInfo() = delete; + SourceLineInfo( char const* _file, std::size_t _line ) noexcept + : file( _file ), + line( _line ) + {} + + SourceLineInfo( SourceLineInfo const& other ) = default; + SourceLineInfo& operator = ( SourceLineInfo const& ) = default; + SourceLineInfo( SourceLineInfo&& ) noexcept = default; + SourceLineInfo& operator = ( SourceLineInfo&& ) noexcept = default; + + bool empty() const noexcept { return file[0] == '\0'; } + bool operator == ( SourceLineInfo const& other ) const noexcept; + bool operator < ( SourceLineInfo const& other ) const noexcept; + + char const* file; + std::size_t line; + }; + + std::ostream& operator << ( std::ostream& os, SourceLineInfo const& info ); + + // Bring in operator<< from global namespace into Catch namespace + // This is necessary because the overload of operator<< above makes + // lookup stop at namespace Catch + using ::operator<<; + + // Use this in variadic streaming macros to allow + // >> +StreamEndStop + // as well as + // >> stuff +StreamEndStop + struct StreamEndStop { + std::string operator+() const; + }; + template + T const& operator + ( T const& value, StreamEndStop ) { + return value; + } +} + +#define CATCH_INTERNAL_LINEINFO \ + ::Catch::SourceLineInfo( __FILE__, static_cast( __LINE__ ) ) + +// end catch_common.h +namespace Catch { + + struct RegistrarForTagAliases { + RegistrarForTagAliases( char const* alias, char const* tag, SourceLineInfo const& lineInfo ); + }; + +} // end namespace Catch + +#define CATCH_REGISTER_TAG_ALIAS( alias, spec ) \ + CATCH_INTERNAL_START_WARNINGS_SUPPRESSION \ + CATCH_INTERNAL_SUPPRESS_GLOBALS_WARNINGS \ + namespace{ Catch::RegistrarForTagAliases INTERNAL_CATCH_UNIQUE_NAME( AutoRegisterTagAlias )( alias, spec, CATCH_INTERNAL_LINEINFO ); } \ + CATCH_INTERNAL_STOP_WARNINGS_SUPPRESSION + +// end catch_tag_alias_autoregistrar.h +// start catch_test_registry.h + +// start catch_interfaces_testcase.h + +#include + +namespace Catch { + + class TestSpec; + + struct ITestInvoker { + virtual void invoke () const = 0; + virtual ~ITestInvoker(); + }; + + 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 isThrowSafe( TestCase const& testCase, IConfig const& config ); + 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 ); + +} + +// end catch_interfaces_testcase.h +// start catch_stringref.h + +#include +#include +#include +#include + +namespace Catch { + + /// A non-owning string class (similar to the forthcoming std::string_view) + /// Note that, because a StringRef may be a substring of another string, + /// it may not be null terminated. + class StringRef { + public: + using size_type = std::size_t; + using const_iterator = const char*; + + private: + static constexpr char const* const s_empty = ""; + + char const* m_start = s_empty; + size_type m_size = 0; + + public: // construction + constexpr StringRef() noexcept = default; + + StringRef( char const* rawChars ) noexcept; + + constexpr StringRef( char const* rawChars, size_type size ) noexcept + : m_start( rawChars ), + m_size( size ) + {} + + StringRef( std::string const& stdString ) noexcept + : m_start( stdString.c_str() ), + m_size( stdString.size() ) + {} + + explicit operator std::string() const { + return std::string(m_start, m_size); + } + + public: // operators + auto operator == ( StringRef const& other ) const noexcept -> bool; + auto operator != (StringRef const& other) const noexcept -> bool { + return !(*this == other); + } + + auto operator[] ( size_type index ) const noexcept -> char { + assert(index < m_size); + return m_start[index]; + } + + public: // named queries + constexpr auto empty() const noexcept -> bool { + return m_size == 0; + } + constexpr auto size() const noexcept -> size_type { + return m_size; + } + + // Returns the current start pointer. If the StringRef is not + // null-terminated, throws std::domain_exception + auto c_str() const -> char const*; + + public: // substrings and searches + // Returns a substring of [start, start + length). + // If start + length > size(), then the substring is [start, size()). + // If start > size(), then the substring is empty. + auto substr( size_type start, size_type length ) const noexcept -> StringRef; + + // Returns the current start pointer. May not be null-terminated. + auto data() const noexcept -> char const*; + + constexpr auto isNullTerminated() const noexcept -> bool { + return m_start[m_size] == '\0'; + } + + public: // iterators + constexpr const_iterator begin() const { return m_start; } + constexpr const_iterator end() const { return m_start + m_size; } + }; + + auto operator += ( std::string& lhs, StringRef const& sr ) -> std::string&; + auto operator << ( std::ostream& os, StringRef const& sr ) -> std::ostream&; + + constexpr auto operator "" _sr( char const* rawChars, std::size_t size ) noexcept -> StringRef { + return StringRef( rawChars, size ); + } +} // namespace Catch + +constexpr auto operator "" _catch_sr( char const* rawChars, std::size_t size ) noexcept -> Catch::StringRef { + return Catch::StringRef( rawChars, size ); +} + +// end catch_stringref.h +// start catch_preprocessor.hpp + + +#define CATCH_RECURSION_LEVEL0(...) __VA_ARGS__ +#define CATCH_RECURSION_LEVEL1(...) CATCH_RECURSION_LEVEL0(CATCH_RECURSION_LEVEL0(CATCH_RECURSION_LEVEL0(__VA_ARGS__))) +#define CATCH_RECURSION_LEVEL2(...) CATCH_RECURSION_LEVEL1(CATCH_RECURSION_LEVEL1(CATCH_RECURSION_LEVEL1(__VA_ARGS__))) +#define CATCH_RECURSION_LEVEL3(...) CATCH_RECURSION_LEVEL2(CATCH_RECURSION_LEVEL2(CATCH_RECURSION_LEVEL2(__VA_ARGS__))) +#define CATCH_RECURSION_LEVEL4(...) CATCH_RECURSION_LEVEL3(CATCH_RECURSION_LEVEL3(CATCH_RECURSION_LEVEL3(__VA_ARGS__))) +#define CATCH_RECURSION_LEVEL5(...) CATCH_RECURSION_LEVEL4(CATCH_RECURSION_LEVEL4(CATCH_RECURSION_LEVEL4(__VA_ARGS__))) + +#ifdef CATCH_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR +#define INTERNAL_CATCH_EXPAND_VARGS(...) __VA_ARGS__ +// MSVC needs more evaluations +#define CATCH_RECURSION_LEVEL6(...) CATCH_RECURSION_LEVEL5(CATCH_RECURSION_LEVEL5(CATCH_RECURSION_LEVEL5(__VA_ARGS__))) +#define CATCH_RECURSE(...) CATCH_RECURSION_LEVEL6(CATCH_RECURSION_LEVEL6(__VA_ARGS__)) +#else +#define CATCH_RECURSE(...) CATCH_RECURSION_LEVEL5(__VA_ARGS__) +#endif + +#define CATCH_REC_END(...) +#define CATCH_REC_OUT + +#define CATCH_EMPTY() +#define CATCH_DEFER(id) id CATCH_EMPTY() + +#define CATCH_REC_GET_END2() 0, CATCH_REC_END +#define CATCH_REC_GET_END1(...) CATCH_REC_GET_END2 +#define CATCH_REC_GET_END(...) CATCH_REC_GET_END1 +#define CATCH_REC_NEXT0(test, next, ...) next CATCH_REC_OUT +#define CATCH_REC_NEXT1(test, next) CATCH_DEFER ( CATCH_REC_NEXT0 ) ( test, next, 0) +#define CATCH_REC_NEXT(test, next) CATCH_REC_NEXT1(CATCH_REC_GET_END test, next) + +#define CATCH_REC_LIST0(f, x, peek, ...) , f(x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST1) ) ( f, peek, __VA_ARGS__ ) +#define CATCH_REC_LIST1(f, x, peek, ...) , f(x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST0) ) ( f, peek, __VA_ARGS__ ) +#define CATCH_REC_LIST2(f, x, peek, ...) f(x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST1) ) ( f, peek, __VA_ARGS__ ) + +#define CATCH_REC_LIST0_UD(f, userdata, x, peek, ...) , f(userdata, x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST1_UD) ) ( f, userdata, peek, __VA_ARGS__ ) +#define CATCH_REC_LIST1_UD(f, userdata, x, peek, ...) , f(userdata, x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST0_UD) ) ( f, userdata, peek, __VA_ARGS__ ) +#define CATCH_REC_LIST2_UD(f, userdata, x, peek, ...) f(userdata, x) CATCH_DEFER ( CATCH_REC_NEXT(peek, CATCH_REC_LIST1_UD) ) ( f, userdata, peek, __VA_ARGS__ ) + +// Applies the function macro `f` to each of the remaining parameters, inserts commas between the results, +// and passes userdata as the first parameter to each invocation, +// e.g. CATCH_REC_LIST_UD(f, x, a, b, c) evaluates to f(x, a), f(x, b), f(x, c) +#define CATCH_REC_LIST_UD(f, userdata, ...) CATCH_RECURSE(CATCH_REC_LIST2_UD(f, userdata, __VA_ARGS__, ()()(), ()()(), ()()(), 0)) + +#define CATCH_REC_LIST(f, ...) CATCH_RECURSE(CATCH_REC_LIST2(f, __VA_ARGS__, ()()(), ()()(), ()()(), 0)) + +#define INTERNAL_CATCH_EXPAND1(param) INTERNAL_CATCH_EXPAND2(param) +#define INTERNAL_CATCH_EXPAND2(...) INTERNAL_CATCH_NO## __VA_ARGS__ +#define INTERNAL_CATCH_DEF(...) INTERNAL_CATCH_DEF __VA_ARGS__ +#define INTERNAL_CATCH_NOINTERNAL_CATCH_DEF +#define INTERNAL_CATCH_STRINGIZE(...) INTERNAL_CATCH_STRINGIZE2(__VA_ARGS__) +#ifndef CATCH_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR +#define INTERNAL_CATCH_STRINGIZE2(...) #__VA_ARGS__ +#define INTERNAL_CATCH_STRINGIZE_WITHOUT_PARENS(param) INTERNAL_CATCH_STRINGIZE(INTERNAL_CATCH_REMOVE_PARENS(param)) +#else +// MSVC is adding extra space and needs another indirection to expand INTERNAL_CATCH_NOINTERNAL_CATCH_DEF +#define INTERNAL_CATCH_STRINGIZE2(...) INTERNAL_CATCH_STRINGIZE3(__VA_ARGS__) +#define INTERNAL_CATCH_STRINGIZE3(...) #__VA_ARGS__ +#define INTERNAL_CATCH_STRINGIZE_WITHOUT_PARENS(param) (INTERNAL_CATCH_STRINGIZE(INTERNAL_CATCH_REMOVE_PARENS(param)) + 1) +#endif + +#define INTERNAL_CATCH_MAKE_NAMESPACE2(...) ns_##__VA_ARGS__ +#define INTERNAL_CATCH_MAKE_NAMESPACE(name) INTERNAL_CATCH_MAKE_NAMESPACE2(name) + +#define INTERNAL_CATCH_REMOVE_PARENS(...) INTERNAL_CATCH_EXPAND1(INTERNAL_CATCH_DEF __VA_ARGS__) + +#ifndef CATCH_CONFIG_TRADITIONAL_MSVC_PREPROCESSOR +#define INTERNAL_CATCH_MAKE_TYPE_LIST2(...) decltype(get_wrapper()) +#define INTERNAL_CATCH_MAKE_TYPE_LIST(...) INTERNAL_CATCH_MAKE_TYPE_LIST2(INTERNAL_CATCH_REMOVE_PARENS(__VA_ARGS__)) +#else +#define INTERNAL_CATCH_MAKE_TYPE_LIST2(...) INTERNAL_CATCH_EXPAND_VARGS(decltype(get_wrapper())) +#define INTERNAL_CATCH_MAKE_TYPE_LIST(...) INTERNAL_CATCH_EXPAND_VARGS(INTERNAL_CATCH_MAKE_TYPE_LIST2(INTERNAL_CATCH_REMOVE_PARENS(__VA_ARGS__))) +#endif + +#define INTERNAL_CATCH_MAKE_TYPE_LISTS_FROM_TYPES(...)\ + CATCH_REC_LIST(INTERNAL_CATCH_MAKE_TYPE_LIST,__VA_ARGS__) + +#define INTERNAL_CATCH_REMOVE_PARENS_1_ARG(_0) INTERNAL_CATCH_REMOVE_PARENS(_0) +#define INTERNAL_CATCH_REMOVE_PARENS_2_ARG(_0, _1) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_1_ARG(_1) +#define INTERNAL_CATCH_REMOVE_PARENS_3_ARG(_0, _1, _2) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_2_ARG(_1, _2) +#define INTERNAL_CATCH_REMOVE_PARENS_4_ARG(_0, _1, _2, _3) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_3_ARG(_1, _2, _3) +#define INTERNAL_CATCH_REMOVE_PARENS_5_ARG(_0, _1, _2, _3, _4) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_4_ARG(_1, _2, _3, _4) +#define INTERNAL_CATCH_REMOVE_PARENS_6_ARG(_0, _1, _2, _3, _4, _5) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_5_ARG(_1, _2, _3, _4, _5) +#define INTERNAL_CATCH_REMOVE_PARENS_7_ARG(_0, _1, _2, _3, _4, _5, _6) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_6_ARG(_1, _2, _3, _4, _5, _6) +#define INTERNAL_CATCH_REMOVE_PARENS_8_ARG(_0, _1, _2, _3, _4, _5, _6, _7) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_7_ARG(_1, _2, _3, _4, _5, _6, _7) +#define INTERNAL_CATCH_REMOVE_PARENS_9_ARG(_0, _1, _2, _3, _4, _5, _6, _7, _8) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_8_ARG(_1, _2, _3, _4, _5, _6, _7, _8) +#define INTERNAL_CATCH_REMOVE_PARENS_10_ARG(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_9_ARG(_1, _2, _3, _4, _5, _6, _7, _8, _9) +#define INTERNAL_CATCH_REMOVE_PARENS_11_ARG(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10) INTERNAL_CATCH_REMOVE_PARENS(_0), INTERNAL_CATCH_REMOVE_PARENS_10_ARG(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10) + +#define INTERNAL_CATCH_VA_NARGS_IMPL(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, N, ...) N + +#define INTERNAL_CATCH_TYPE_GEN\ + template struct TypeList {};\ + template\ + constexpr auto get_wrapper() noexcept -> TypeList { return {}; }\ + template class...> struct TemplateTypeList{};\ + template class...Cs>\ + constexpr auto get_wrapper() noexcept -> TemplateTypeList { return {}; }\ + template\ + struct append;\ + template\ + struct rewrap;\ + template class, typename...>\ + struct create;\ + template class, typename>\ + struct convert;\ + \ + template \ + struct append { using type = T; };\ + template< template class L1, typename...E1, template class L2, typename...E2, typename...Rest>\ + struct append, L2, Rest...> { using type = typename append, Rest...>::type; };\ + template< template class L1, typename...E1, typename...Rest>\ + struct append, TypeList, Rest...> { using type = L1; };\ + \ + template< template class Container, template class List, typename...elems>\ + struct rewrap, List> { using type = TypeList>; };\ + template< template class Container, template class List, class...Elems, typename...Elements>\ + struct rewrap, List, Elements...> { using type = typename append>, typename rewrap, Elements...>::type>::type; };\ + \ + template