# Distributed under the OSI-approved BSD 3-Clause License. See accompanying # file Copyright.txt or https://cmake.org/licensing for details. function(add_command NAME) set(_args "") # use ARGV* instead of ARGN, because ARGN splits arrays into multiple arguments math(EXPR _last_arg ${ARGC}-1) foreach(_n RANGE 1 ${_last_arg}) set(_arg "${ARGV${_n}}") if(_arg MATCHES "[^-./:a-zA-Z0-9_]") set(_args "${_args} [==[${_arg}]==]") # form a bracket_argument else() set(_args "${_args} ${_arg}") endif() endforeach() set(script "${script}${NAME}(${_args})\n" PARENT_SCOPE) endfunction() function(catch_discover_tests_impl) cmake_parse_arguments( "" "" "TEST_EXECUTABLE;TEST_WORKING_DIR;TEST_OUTPUT_DIR;TEST_OUTPUT_PREFIX;TEST_OUTPUT_SUFFIX;TEST_PREFIX;TEST_REPORTER;TEST_SPEC;TEST_SUFFIX;TEST_LIST;CTEST_FILE" "TEST_EXTRA_ARGS;TEST_PROPERTIES;TEST_EXECUTOR;TEST_DL_PATHS;TEST_DL_FRAMEWORK_PATHS;ADD_TAGS_AS_LABELS" ${ARGN} ) set(add_tags "${_ADD_TAGS_AS_LABELS}") set(prefix "${_TEST_PREFIX}") set(suffix "${_TEST_SUFFIX}") set(spec ${_TEST_SPEC}) set(extra_args ${_TEST_EXTRA_ARGS}) set(properties ${_TEST_PROPERTIES}) set(reporter ${_TEST_REPORTER}) set(output_dir ${_TEST_OUTPUT_DIR}) set(output_prefix ${_TEST_OUTPUT_PREFIX}) set(output_suffix ${_TEST_OUTPUT_SUFFIX}) set(dl_paths ${_TEST_DL_PATHS}) set(dl_framework_paths ${_TEST_DL_FRAMEWORK_PATHS}) set(environment_modifications "") set(script) set(suite) set(tests) if(WIN32) set(dl_paths_variable_name PATH) elseif(APPLE) set(dl_paths_variable_name DYLD_LIBRARY_PATH) else() set(dl_paths_variable_name LD_LIBRARY_PATH) endif() # Run test executable to get list of available tests if(NOT EXISTS "${_TEST_EXECUTABLE}") message(FATAL_ERROR "Specified test executable '${_TEST_EXECUTABLE}' does not exist" ) endif() if(dl_paths) cmake_path(CONVERT "$ENV{${dl_paths_variable_name}}" TO_NATIVE_PATH_LIST env_dl_paths) list(PREPEND env_dl_paths "${dl_paths}") cmake_path(CONVERT "${env_dl_paths}" TO_NATIVE_PATH_LIST paths) set(ENV{${dl_paths_variable_name}} "${paths}") endif() if(APPLE AND dl_framework_paths) cmake_path(CONVERT "$ENV{DYLD_FRAMEWORK_PATH}" TO_NATIVE_PATH_LIST env_dl_framework_paths) list(PREPEND env_dl_framework_paths "${dl_framework_paths}") cmake_path(CONVERT "${env_dl_framework_paths}" TO_NATIVE_PATH_LIST paths) set(ENV{DYLD_FRAMEWORK_PATH} "${paths}") endif() execute_process( COMMAND ${_TEST_EXECUTOR} "${_TEST_EXECUTABLE}" ${spec} --list-tests --reporter json OUTPUT_VARIABLE listing_output RESULT_VARIABLE result WORKING_DIRECTORY "${_TEST_WORKING_DIR}" ) if(NOT ${result} EQUAL 0) message(FATAL_ERROR "Error listing tests from executable '${_TEST_EXECUTABLE}':\n" " Result: ${result}\n" " Output: ${listing_output}\n" ) endif() # Prepare reporter if(reporter) set(reporter_arg "--reporter ${reporter}") # Run test executable to check whether reporter is available # note that the use of --list-reporters is not the important part, # we only want to check whether the execution succeeds with ${reporter_arg} execute_process( COMMAND ${_TEST_EXECUTOR} "${_TEST_EXECUTABLE}" ${spec} ${reporter_arg} --list-reporters OUTPUT_VARIABLE reporter_check_output RESULT_VARIABLE reporter_check_result WORKING_DIRECTORY "${_TEST_WORKING_DIR}" ) if(${reporter_check_result} EQUAL 255) message(FATAL_ERROR "\"${reporter}\" is not a valid reporter!\n" ) elseif(NOT ${reporter_check_result} EQUAL 0) message(FATAL_ERROR "Error checking for reporter in test executable '${_TEST_EXECUTABLE}':\n" " Result: ${reporter_check_result}\n" " Output: ${reporter_check_output}\n" ) endif() endif() # Prepare output dir if(output_dir AND NOT IS_ABSOLUTE ${output_dir}) set(output_dir "${_TEST_WORKING_DIR}/${output_dir}") if(NOT EXISTS ${output_dir}) file(MAKE_DIRECTORY ${output_dir}) endif() endif() if(dl_paths) foreach(path ${dl_paths}) cmake_path(NATIVE_PATH path native_path) list(PREPEND environment_modifications "${dl_paths_variable_name}=path_list_prepend:${native_path}") endforeach() endif() if(APPLE AND dl_framework_paths) foreach(path ${dl_framework_paths}) cmake_path(NATIVE_PATH path native_path) list(PREPEND environment_modifications "DYLD_FRAMEWORK_PATH=path_list_prepend:${native_path}") endforeach() endif() # Parse JSON output for list of tests/class names/tags string(JSON version GET "${listing_output}" "version") if (NOT version STREQUAL "1") message(FATAL_ERROR "Unsupported catch output version: '${version}'") endif() # Speed-up reparsing by cutting away unneeded parts of JSON. string(JSON test_listing GET "${listing_output}" "listings" "tests") string(JSON num_tests LENGTH "${test_listing}") # CMake's foreach-RANGE is inclusive, so we have to subtract 1 math(EXPR num_tests "${num_tests} - 1") foreach(idx RANGE ${num_tests}) string(JSON single_test GET ${test_listing} ${idx}) string(JSON test_tags GET "${single_test}" "tags") string(JSON plain_name GET "${single_test}" "name") # Escape characters in test case names that would be parsed by Catch2 # Note that the \ escaping must happen FIRST! Do not change the order. set(escaped_name "${plain_name}") foreach(char \\ , [ ] ;) string(REPLACE ${char} "\\${char}" escaped_name "${escaped_name}") endforeach(char) # ...add output dir if(output_dir) string(REGEX REPLACE "[^A-Za-z0-9_]" "_" escaped_name_clean "${escaped_name}") set(output_dir_arg "--out ${output_dir}/${output_prefix}${escaped_name_clean}${output_suffix}") endif() # ...and add to script add_command(add_test "${prefix}${plain_name}${suffix}" ${_TEST_EXECUTOR} "${_TEST_EXECUTABLE}" "${escaped_name}" ${extra_args} "${reporter_arg}" "${output_dir_arg}" ) add_command(set_tests_properties "${prefix}${plain_name}${suffix}" PROPERTIES WORKING_DIRECTORY "${_TEST_WORKING_DIR}" ${properties} ) if (add_tags) string(JSON num_tags LENGTH "${test_tags}") math(EXPR num_tags "${num_tags} - 1") set(tag_list "") if (num_tags GREATER_EQUAL "0") foreach(tag_idx RANGE ${num_tags}) string(JSON a_tag GET "${test_tags}" "${tag_idx}") # Catch2's tags can contain semicolons, which are list element separators # in CMake, so we have to escape them. Ideally we could use the [=[...]=] # syntax for this, but CTest currently keeps the square quotes in the label # name. So we add 2 backslashes to escape it instead. # **IMPORTANT**: The number of backslashes depends on how many layers # of CMake the tag goes. If this script is changed, the # number of backslashes to escape may change as well. string(REPLACE ";" "\\;" a_tag "${a_tag}") list(APPEND tag_list "${a_tag}") endforeach() add_command(set_tests_properties "${prefix}${plain_name}${suffix}" PROPERTIES LABELS "${tag_list}" ) endif() endif(add_tags) if(environment_modifications) add_command(set_tests_properties "${prefix}${plain_name}${suffix}" PROPERTIES ENVIRONMENT_MODIFICATION "${environment_modifications}") endif() list(APPEND tests "${prefix}${plain_name}${suffix}") endforeach() # Create a list of all discovered tests, which users may use to e.g. set # properties on the tests add_command(set ${_TEST_LIST} ${tests}) # Write CTest script file(WRITE "${_CTEST_FILE}" "${script}") endfunction() if(CMAKE_SCRIPT_MODE_FILE) catch_discover_tests_impl( TEST_EXECUTABLE ${TEST_EXECUTABLE} TEST_EXECUTOR ${TEST_EXECUTOR} TEST_WORKING_DIR ${TEST_WORKING_DIR} TEST_SPEC ${TEST_SPEC} TEST_EXTRA_ARGS ${TEST_EXTRA_ARGS} TEST_PROPERTIES ${TEST_PROPERTIES} TEST_PREFIX ${TEST_PREFIX} TEST_SUFFIX ${TEST_SUFFIX} TEST_LIST ${TEST_LIST} TEST_REPORTER ${TEST_REPORTER} TEST_OUTPUT_DIR ${TEST_OUTPUT_DIR} TEST_OUTPUT_PREFIX ${TEST_OUTPUT_PREFIX} TEST_OUTPUT_SUFFIX ${TEST_OUTPUT_SUFFIX} TEST_DL_PATHS ${TEST_DL_PATHS} TEST_DL_FRAMEWORK_PATHS ${TEST_DL_FRAMEWORK_PATHS} CTEST_FILE ${CTEST_FILE} ADD_TAGS_AS_LABELS ${ADD_TAGS_AS_LABELS} ) endif()