| # Distributed under the OSI-approved BSD 3-Clause License. See accompanying |
| # file LICENSE.rst or https://cmake.org/licensing for details. |
| |
| cmake_minimum_required(VERSION 3.30) |
| cmake_policy(SET CMP0174 NEW) # TODO: Remove this when we can update the above to 3.31 |
| |
| function(add_command name test_name) |
| set(args "") |
| foreach(arg ${ARGN}) |
| if(arg MATCHES "[^-./:a-zA-Z0-9_]") |
| string(APPEND args " [==[${arg}]==]") |
| else() |
| string(APPEND args " ${arg}") |
| endif() |
| endforeach() |
| string(APPEND script "${name}(${test_name} ${args})\n") |
| set(script "${script}" PARENT_SCOPE) |
| endfunction() |
| |
| function(generate_testname_guards output open_guard_var close_guard_var) |
| set(open_guard "[=[") |
| set(close_guard "]=]") |
| set(counter 1) |
| while("${output}" MATCHES "${close_guard}") |
| math(EXPR counter "${counter} + 1") |
| string(REPEAT "=" ${counter} equals) |
| set(open_guard "[${equals}[") |
| set(close_guard "]${equals}]") |
| endwhile() |
| set(${open_guard_var} "${open_guard}" PARENT_SCOPE) |
| set(${close_guard_var} "${close_guard}" PARENT_SCOPE) |
| endfunction() |
| |
| function(escape_square_brackets output bracket placeholder placeholder_var output_var) |
| if("${output}" MATCHES "\\${bracket}") |
| set(placeholder "${placeholder}") |
| while("${output}" MATCHES "${placeholder}") |
| set(placeholder "${placeholder}_") |
| endwhile() |
| string(REPLACE "${bracket}" "${placeholder}" output "${output}") |
| set(${placeholder_var} "${placeholder}" PARENT_SCOPE) |
| set(${output_var} "${output}" PARENT_SCOPE) |
| endif() |
| endfunction() |
| |
| macro(write_test_to_file) |
| # Store the gtest test name before messing with these strings |
| set(gtest_name ${current_test_suite}.${current_test_name}) |
| |
| set(pretty_test_suite ${current_test_suite}) |
| set(pretty_test_name ${current_test_name}) |
| |
| # Handle disabled tests |
| set(maybe_DISABLED "") |
| if(pretty_test_suite MATCHES "^DISABLED_" OR pretty_test_name MATCHES "^DISABLED_") |
| set(maybe_DISABLED DISABLED YES) |
| string(REGEX REPLACE "^DISABLED_" "" pretty_test_suite "${pretty_test_suite}") |
| string(REGEX REPLACE "^DISABLED_" "" pretty_test_name "${pretty_test_name}") |
| endif() |
| |
| if (NOT current_test_value_param STREQUAL "" AND NOT arg_NO_PRETTY_VALUES) |
| # Remove value param name, if any, from test name |
| string(REGEX REPLACE "^(.+)/.+$" "\\1" pretty_test_name "${pretty_test_name}") |
| set(pretty_test_name "${pretty_test_name}/${current_test_value_param}") |
| endif() |
| |
| if(NOT current_test_type_param STREQUAL "") |
| # Parse type param name from suite name |
| if(pretty_test_suite MATCHES "^(.+)/(.+)$") |
| set(pretty_test_suite "${CMAKE_MATCH_1}") |
| set(current_type_param_name "${CMAKE_MATCH_2}") |
| else() |
| set(current_type_param_name "") |
| endif() |
| if (NOT arg_NO_PRETTY_TYPES) |
| string(APPEND pretty_test_name "<${current_test_type_param}>") |
| elseif(NOT current_type_param_name STREQUAL "") |
| string(APPEND pretty_test_name "<${current_type_param_name}>") |
| endif() |
| endif() |
| |
| set(test_name_template "@prefix@@pretty_test_suite@.@pretty_test_name@@suffix@") |
| string(CONFIGURE "${test_name_template}" testname) |
| |
| if(NOT "${arg_TEST_XML_OUTPUT_DIR}" STREQUAL "") |
| set(TEST_XML_OUTPUT_PARAM "--gtest_output=xml:${arg_TEST_XML_OUTPUT_DIR}/${prefix}${gtest_name}${suffix}.xml") |
| else() |
| set(TEST_XML_OUTPUT_PARAM "") |
| endif() |
| |
| # unescape [] |
| if(open_sb) |
| string(REPLACE "${open_sb}" "[" testname "${testname}") |
| endif() |
| if(close_sb) |
| string(REPLACE "${close_sb}" "]" testname "${testname}") |
| endif() |
| set(guarded_testname "${open_guard}${testname}${close_guard}") |
| # Add to script. Do not use add_command() here because it messes up the |
| # handling of empty values when forwarding arguments, and we need to |
| # preserve those carefully for arg_TEST_EXECUTOR and arg_EXTRA_ARGS. |
| string(APPEND script "add_test(${guarded_testname} ${launcherArgs}") |
| foreach(arg IN ITEMS |
| "${arg_TEST_EXECUTABLE}" |
| "--gtest_filter=${gtest_name}" |
| "--gtest_also_run_disabled_tests" |
| ${TEST_XML_OUTPUT_PARAM} |
| ) |
| |
| if(arg MATCHES "[^-./:a-zA-Z0-9_]") |
| string(APPEND script " [==[${arg}]==]") |
| else() |
| string(APPEND script " ${arg}") |
| endif() |
| endforeach() |
| |
| if(arg_TEST_EXTRA_ARGS) |
| list(JOIN arg_TEST_EXTRA_ARGS "]==] [==[" extra_args) |
| string(APPEND script " [==[${extra_args}]==]") |
| endif() |
| string(APPEND script ")\n") |
| |
| set(maybe_LOCATION "") |
| if(NOT current_test_file STREQUAL "" AND NOT current_test_line STREQUAL "") |
| set(maybe_LOCATION DEF_SOURCE_LINE "${current_test_file}:${current_test_line}") |
| endif() |
| |
| add_command(set_tests_properties |
| "${guarded_testname}" |
| PROPERTIES |
| ${maybe_DISABLED} |
| ${maybe_LOCATION} |
| WORKING_DIRECTORY "${arg_TEST_WORKING_DIR}" |
| SKIP_REGULAR_EXPRESSION "\\[ SKIPPED \\]" |
| ${arg_TEST_PROPERTIES} |
| ) |
| |
| # possibly unbalanced square brackets render lists invalid so skip such |
| # tests in ${arg_TEST_LIST} |
| if(NOT "${testname}" MATCHES [=[(\[|\])]=]) |
| # escape ; |
| string(REPLACE [[;]] [[\\;]] testname "${testname}") |
| list(APPEND tests_buffer "${testname}") |
| list(LENGTH tests_buffer tests_buffer_length) |
| if(tests_buffer_length GREATER "250") |
| # Chunk updates to the final "tests" variable, keeping the |
| # "tests_buffer" variable that we append each test to relatively |
| # small. This mitigates worsening performance impacts for the |
| # corner case of having many thousands of tests. |
| list(APPEND tests "${tests_buffer}") |
| set(tests_buffer "") |
| endif() |
| endif() |
| |
| # If we've built up a sizable script so far, write it out as a chunk now |
| # so we don't accumulate a massive string to write at the end |
| string(LENGTH "${script}" script_len) |
| if(${script_len} GREATER "50000") |
| file(APPEND "${arg_CTEST_FILE}" "${script}") |
| set(script "") |
| endif() |
| endmacro() |
| |
| macro(parse_tests_from_output) |
| generate_testname_guards("${output}" open_guard close_guard) |
| escape_square_brackets("${output}" "[" "__osb" open_sb output) |
| escape_square_brackets("${output}" "]" "__csb" close_sb output) |
| |
| # Preserve semicolon in test-parameters |
| string(REPLACE [[;]] [[\;]] output "${output}") |
| string(REPLACE "\n" ";" output "${output}") |
| |
| # Command line output doesn't contain information about the file and line number of the tests |
| set(current_test_file "") |
| set(current_test_line "") |
| |
| # Parse output |
| foreach(line ${output}) |
| # Skip header |
| if(line MATCHES "gtest_main\\.cc") |
| continue() |
| endif() |
| |
| if(line STREQUAL "") |
| continue() |
| endif() |
| |
| # Do we have a module name or a test name? |
| if(NOT line MATCHES "^ ") |
| set(current_test_type_param "") |
| |
| # Module; remove trailing '.' to get just the name... |
| string(REGEX REPLACE "\\.( *#.*)?$" "" current_test_suite "${line}") |
| if(line MATCHES "# *TypeParam = (.*)$") |
| set(current_test_type_param "${CMAKE_MATCH_1}") |
| endif() |
| else() |
| string(STRIP "${line}" test) |
| string(REGEX REPLACE " ( *#.*)?$" "" current_test_name "${test}") |
| |
| set(current_test_value_param "") |
| if(line MATCHES "# *GetParam\\(\\) = (.*)$") |
| set(current_test_value_param "${CMAKE_MATCH_1}") |
| endif() |
| |
| write_test_to_file() |
| endif() |
| endforeach() |
| endmacro() |
| |
| macro(get_json_member_with_default json_variable member_name out_variable) |
| string(JSON ${out_variable} |
| ERROR_VARIABLE error_param |
| GET "${${json_variable}}" "${member_name}" |
| ) |
| if(error_param) |
| # Member not present |
| set(${out_variable} "") |
| endif() |
| endmacro() |
| |
| macro(parse_tests_from_json json_file) |
| if(NOT EXISTS "${json_file}") |
| message(FATAL_ERROR "Missing expected JSON file with test list: ${json_file}") |
| endif() |
| |
| file(READ "${json_file}" test_json) |
| string(JSON test_suites_json GET "${test_json}" "testsuites") |
| |
| # Return if there are no testsuites |
| string(JSON len_test_suites LENGTH "${test_suites_json}") |
| if(len_test_suites LESS_EQUAL 0) |
| return() |
| endif() |
| |
| set(open_sb) |
| set(close_sb) |
| |
| math(EXPR upper_limit_test_suite_range "${len_test_suites} - 1") |
| |
| foreach(index_test_suite RANGE ${upper_limit_test_suite_range}) |
| string(JSON test_suite_json GET "${test_suites_json}" ${index_test_suite}) |
| |
| # "suite" is expected to be set in write_test_to_file(). When parsing the |
| # plain text output, "suite" is expected to be the original suite name |
| # before accounting for pretty names. This may be used to construct the |
| # name of XML output results files. |
| string(JSON current_test_suite GET "${test_suite_json}" "name") |
| string(JSON tests_json GET "${test_suite_json}" "testsuite") |
| |
| # Skip test suites without tests |
| string(JSON len_tests LENGTH "${tests_json}") |
| if(len_tests LESS_EQUAL 0) |
| continue() |
| endif() |
| |
| math(EXPR upper_limit_test_range "${len_tests} - 1") |
| foreach(index_test RANGE ${upper_limit_test_range}) |
| string(JSON test_json GET "${tests_json}" ${index_test}) |
| |
| string(JSON len_test_parameters LENGTH "${test_json}") |
| if(len_test_parameters LESS_EQUAL 0) |
| continue() |
| endif() |
| |
| get_json_member_with_default(test_json "name" current_test_name) |
| get_json_member_with_default(test_json "file" current_test_file) |
| get_json_member_with_default(test_json "line" current_test_line) |
| get_json_member_with_default(test_json "value_param" current_test_value_param) |
| get_json_member_with_default(test_json "type_param" current_test_type_param) |
| |
| generate_testname_guards( |
| "${current_test_suite}${current_test_name}${current_test_value_param}${current_test_type_param}" |
| open_guard close_guard |
| ) |
| write_test_to_file() |
| endforeach() |
| endforeach() |
| endmacro() |
| |
| function(gtest_discover_tests_impl) |
| |
| set(options "") |
| set(oneValueArgs |
| NO_PRETTY_TYPES # These two take a value, unlike gtest_discover_tests() |
| NO_PRETTY_VALUES # |
| TEST_EXECUTABLE |
| TEST_WORKING_DIR |
| TEST_PREFIX |
| TEST_SUFFIX |
| TEST_LIST |
| CTEST_FILE |
| TEST_DISCOVERY_TIMEOUT |
| TEST_XML_OUTPUT_DIR |
| # The following are all multi-value arguments in gtest_discover_tests(), |
| # but they are each given to us as a single argument. We parse them that |
| # way to avoid problems with preserving empty list values and escaping. |
| TEST_FILTER |
| TEST_EXTRA_ARGS |
| TEST_DISCOVERY_EXTRA_ARGS |
| TEST_PROPERTIES |
| TEST_EXECUTOR |
| ) |
| set(multiValueArgs "") |
| cmake_parse_arguments(PARSE_ARGV 0 arg |
| "${options}" "${oneValueArgs}" "${multiValueArgs}" |
| ) |
| |
| set(prefix "${arg_TEST_PREFIX}") |
| set(suffix "${arg_TEST_SUFFIX}") |
| set(script) |
| set(tests) |
| set(tests_buffer "") |
| |
| # If a file at ${arg_CTEST_FILE} already exists, we overwrite it. |
| file(REMOVE "${arg_CTEST_FILE}") |
| |
| set(filter) |
| if(arg_TEST_FILTER) |
| set(filter "--gtest_filter=${arg_TEST_FILTER}") |
| endif() |
| |
| # CMP0178 has already been handled in gtest_discover_tests(), so we only need |
| # to implement NEW behavior here. This means preserving empty arguments for |
| # TEST_EXECUTOR. For OLD or WARN, gtest_discover_tests() already removed any |
| # empty arguments. |
| set(launcherArgs "") |
| if(NOT "${arg_TEST_EXECUTOR}" STREQUAL "") |
| list(JOIN arg_TEST_EXECUTOR "]==] [==[" launcherArgs) |
| set(launcherArgs "[==[${launcherArgs}]==]") |
| endif() |
| |
| # Run test executable to get list of available tests |
| if(NOT EXISTS "${arg_TEST_EXECUTABLE}") |
| message(FATAL_ERROR |
| "Specified test executable does not exist.\n" |
| " Path: '${arg_TEST_EXECUTABLE}'" |
| ) |
| endif() |
| |
| set(discovery_extra_args "") |
| if(NOT "${arg_TEST_DISCOVERY_EXTRA_ARGS}" STREQUAL "") |
| list(JOIN arg_TEST_DISCOVERY_EXTRA_ARGS "]==] [==[" discovery_extra_args) |
| set(discovery_extra_args "[==[${discovery_extra_args}]==]") |
| endif() |
| |
| set(json_file "${arg_TEST_WORKING_DIR}/cmake_test_discovery.json") |
| |
| # Remove json file to make sure we don't pick up an outdated one |
| file(REMOVE "${json_file}") |
| |
| cmake_language(EVAL CODE |
| "execute_process( |
| COMMAND ${launcherArgs} [==[${arg_TEST_EXECUTABLE}]==] |
| --gtest_list_tests |
| [==[--gtest_output=json:${json_file}]==] |
| ${filter} |
| ${discovery_extra_args} |
| WORKING_DIRECTORY [==[${arg_TEST_WORKING_DIR}]==] |
| TIMEOUT ${arg_TEST_DISCOVERY_TIMEOUT} |
| OUTPUT_VARIABLE output |
| RESULT_VARIABLE result |
| )" |
| ) |
| |
| if(NOT ${result} EQUAL 0) |
| string(REPLACE "\n" "\n " output "${output}") |
| if(arg_TEST_EXECUTOR) |
| set(path "${arg_TEST_EXECUTOR} ${arg_TEST_EXECUTABLE}") |
| else() |
| set(path "${arg_TEST_EXECUTABLE}") |
| endif() |
| message(FATAL_ERROR |
| "Error running test executable.\n" |
| " Path: '${path}'\n" |
| " Working directory: '${arg_TEST_WORKING_DIR}'\n" |
| " Result: ${result}\n" |
| " Output:\n" |
| " ${output}\n" |
| ) |
| endif() |
| |
| if(EXISTS "${json_file}") |
| parse_tests_from_json("${json_file}") |
| else() |
| # gtest < 1.8.1, and all gtest compiled with GTEST_HAS_FILE_SYSTEM=0, don't |
| # recognize the --gtest_output=json option, and issue a warning or error on |
| # stdout about it being unrecognized, but still return an exit code 0 for |
| # success. All versions report the test list on stdout whether |
| # --gtest_output=json is recognized or not. |
| |
| # NOTE: Because we are calling a macro, we don't want to pass "output" as |
| # an argument because it messes up the contents passed through due to the |
| # different escaping, etc. that gets applied. We rely on it picking up the |
| # "output" variable we have already set here. |
| parse_tests_from_output() |
| endif() |
| |
| if(NOT tests_buffer STREQUAL "") |
| list(APPEND tests "${tests_buffer}") |
| endif() |
| |
| # Create a list of all discovered tests, which users may use to e.g. set |
| # properties on the tests |
| add_command(set "" ${arg_TEST_LIST} "${tests}") |
| |
| # Write remaining content to the CTest script |
| file(APPEND "${arg_CTEST_FILE}" "${script}") |
| endfunction() |
| |
| if(CMAKE_SCRIPT_MODE_FILE) |
| gtest_discover_tests_impl( |
| NO_PRETTY_TYPES ${NO_PRETTY_TYPES} |
| NO_PRETTY_VALUES ${NO_PRETTY_VALUES} |
| TEST_EXECUTABLE ${TEST_EXECUTABLE} |
| TEST_EXECUTOR "${TEST_EXECUTOR}" |
| TEST_WORKING_DIR ${TEST_WORKING_DIR} |
| TEST_PREFIX ${TEST_PREFIX} |
| TEST_SUFFIX ${TEST_SUFFIX} |
| TEST_FILTER ${TEST_FILTER} |
| TEST_LIST ${TEST_LIST} |
| CTEST_FILE ${CTEST_FILE} |
| TEST_DISCOVERY_TIMEOUT ${TEST_DISCOVERY_TIMEOUT} |
| TEST_XML_OUTPUT_DIR ${TEST_XML_OUTPUT_DIR} |
| TEST_EXTRA_ARGS "${TEST_EXTRA_ARGS}" |
| TEST_DISCOVERY_EXTRA_ARGS "${TEST_DISCOVERY_EXTRA_ARGS}" |
| TEST_PROPERTIES "${TEST_PROPERTIES}" |
| ) |
| endif() |