GoogleTest CMake 和 Make 测试未运行

Pat*_*cia 5 c++ linux makefile cmake googletest

我承认我有一个独特的情况。我们使用 Make 构建我们的应用程序。但是我的 IDE CLion 使用 CMake。所以我尝试将 GoogleTest 设置为在两者上运行(有点)。我可以两种方式编译我的代码(在命令行中使用 make 并从我的 IDE 中构建)。但是当我选择测试装置并单击运行按钮时,在 CLion 中,没有找到测试,这就是我收到的:

Running main() from gtest_main.cc  
[==========] Running 0 tests from 0 test cases.  
[==========] 0 tests from 0 test cases ran. (0 ms total)  
[ PASSED  ] 0 tests.

Process finished with exit code 0
Run Code Online (Sandbox Code Playgroud)

这是我的测试夹具:

#include <gtest/gtest.h>
#include "OPProperties.h"

namespace {
// The fixture for testing class OPPropertiesTestTest.
    class OPPropertiesTestTest : public ::testing::Test {
    protected:
        // You can remove any or all of the following functions if its body
        // is empty.

        OPPropertiesTestTest() {
            // You can do set-up work for each test here.
        }

        virtual ~OPPropertiesTestTest() {
            // You can do clean-up work that doesn't throw exceptions here.
        }

        // If the constructor and destructor are not enough for setting up
        // and cleaning up each test, you can define the following methods:

        virtual void SetUp() {
            // Code here will be called immediately after the constructor (right
            // before each test).
        }

        virtual void TearDown() {
            // Code here will be called immediately after each test (right
            // before the destructor).
        }

        // Objects declared here can be used by all tests in the test case for OPPropertiesTestTest.
    };

    TEST_F(OPPropertiesTestTest, ThisTestWillPass) {
        EXPECT_EQ(0, 0);
    }

    TEST_F(OPPropertiesTestTest, ThisTestWillFail) {
        EXPECT_EQ(0, 5);
    }


}  // namespace


int main(int argc, char **argv) {
    ::testing::InitGoogleTest(&argc, argv);
    return RUN_ALL_TESTS();
}
Run Code Online (Sandbox Code Playgroud)

这是我的 CMakeLists.txt 文件:

cmake_minimum_required(VERSION 2.8)
project(oneprint)

set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -pthread -lgtest")
add_definitions(-DBOOST_LOG_DYN_LINK)


set(SOURCE_FILES
    src/controllers/OPProperties.cpp
    src/controllers/OPProperties.h
    src/main.cpp)

include_directories(src/controllers)

set(BOOST_ROOT "/usr/local/lib")
set(Boost_USE_STATIC_LIBS OFF)
set(Boost_USE_MULTITHREADED ON)
unset(Boost_INCLUDE_DIR CACHE)
unset(Boost_LIBRARY_DIRS CACHE)
#set(Boost_LIBRARY_DIR /usr/local/arm/lib)
set(OpenCV_LIBRARY_DIR /usr/include/opencv2)
set(Innovatrics_LIBRARY_DIR /usr/local/arm/lib)

find_package(OpenCV REQUIRED)

find_package(Boost 1.57.0 COMPONENTS filesystem thread log chrono system atomic program_options REQUIRED)
include_directories(${Boost_INCLUDE_DIRS})

enable_testing()
find_package(GTest REQUIRED)
include_directories(${GTEST_INCLUDE_DIRS})

add_executable(OnePrint ${SOURCE_FILES})

target_link_libraries(OnePrint ${OpenCV_LIBS})
target_link_libraries(OnePrint ${Boost_LIBRARIES})
target_link_libraries(OnePrint ${Innovatrics_LIBRARY_DIR})
target_link_libraries(OnePrint ${GTEST_BOTH_LIBRARIES})
target_link_libraries(OnePrint pthread)
Run Code Online (Sandbox Code Playgroud)

我在 src 下添加了一个名为 tests 的文件夹,这是我的测试装置 OPPropertiesTestTest 所在的位置。我还在顶层添加了一个测试文件夹。此文件夹中包含一个 Makefile 和一个 Srcs.mak 文件。

这是生成文件:

TARGET = oneprint
BASE = ../

-include $(BASE)Defs.x86.mak
-include $(BASE)OpenCV.mak
-include $(BASE)Boost.mak
-include $(BASE)Innovatrics.mak
-include $(BASE)GTest.mak

-include $(BASE)Incl.mak
-include Srcs.mak

-include $(BASE)Common.mak
-include $(BASE)App.mak
Run Code Online (Sandbox Code Playgroud)

这是 Srcs.mak 文件:

VPATH = \
../src/controllers:\
../src:\
../src/tests

CPP_SRCS = \
OPProperties.cpp \

# test files
CPP_SRCS += \
OPPropertiesTest.cpp
Run Code Online (Sandbox Code Playgroud)

Fra*_*ser 5

我通常不希望看到 makefile 被混合到 CMake 管理的项目中;听起来你的设置很复杂?

除此之外,我认为这里的根本原因可能是双重的。我认为/src/tests/实际上并没有构建测试可执行文件。我在你的 CMakelists.txt 中看不到任何会导致它被构建的东西,所以除非你做了一些你没有向我们展示的额外内容,否则不会编译测试文件。

可能使它们看起来像的是,您正在将 gtest 提供的帮助程序库链接到您的OnePrint目标中。这个助手库的不寻常之处在于它实现了一个main()功能来保存用户必须指定自己的功能。

你正在做这行

target_link_libraries(OnePrint ${GTEST_BOTH_LIBRARIES})
Run Code Online (Sandbox Code Playgroud)

来自文档的 GTEST_BOTH_LIBRARIES是一个包含 libgtest 和 libgtest-main 的变量。您只需要 libgtest,因为您已经编写了自己的main(). 所以你应该GTEST_LIBRARIES改用。


注意:CMake 代码还有一些其他问题:

set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -pthread -lgtest")
Run Code Online (Sandbox Code Playgroud)

这不是跨平台的(这对您来说可能无关紧要),例如 MSVC 不会识别任何这些标志。如果确实重要,您需要将其包装在if(UNIX)orif(NOT MSVC)块中。

此外,这会全局设置编译器标志,即为项目中定义的每个目标设置。如果您需要更细粒度的控制,请查看target_compile_options哪个可以让您为每个目标指定这些。

另一个问题是-lgtest零件。这实际上是调用链接器来链接 gtest 库。但是,CMake 会为您处理这些问题 - 您在调用target_link_libraries(OnePrint ${GTEST_LIBRARIES}).


add_definitions(-DBOOST_LOG_DYN_LINK)
Run Code Online (Sandbox Code Playgroud)

虽然这很好,但关于在全球范围内应用的相同评论也适用。每个目标的等价物是target_compile_definitions


set(BOOST_ROOT "/usr/local/lib")
Run Code Online (Sandbox Code Playgroud)

这可能是 Boost 在机器上的位置,但对于其他开发人员来说可能并非如此。该变量实际上是由单个用户在调用 CMake 时指定的(如果需要……CMake 会自动搜索许多路径)——您不应该将这样的路径硬编码到 CMake 文件中。


target_link_libraries(OnePrint pthread)
Run Code Online (Sandbox Code Playgroud)

同样,pthread如果您使用 MSVC ,则不适合链接;这可能应该包裹在一个if块中。


最后一点是您可以在target_link_libraries命令中指定多个依赖项,因此您可以将它们更改为:

if(NOT MSVC)
  set(PThreadLib pthread)
endif()
target_link_libraries(OnePrint
    ${OpenCV_LIBS}
    ${Boost_LIBRARIES}
    ${Innovatrics_LIBRARY_DIR}
    ${GTEST_LIBRARIES}
    ${PThreadLib})
Run Code Online (Sandbox Code Playgroud)