Suppose we've written an application library in Unison, We could do some test to make sure that it works as expected.
In this article, we'll walk through writing a simple testcase in Unison with
googletest.
What is googletest?
- Google's C++ test framework. https://github.com/google/googletest
- You'll find the document here https://github.com/google/googletest/tree/master/googletest/docs.
- It will be a good start to read this https://github.com/google/googletest/blob/master/googletest/docs/primer.md.
Build googletest as a shared library
First of all, to work in Unison, we need to build googletest into a shared library.
Choose the right version of googletest
Starting from version v1.10.0, googletest requires the compiler support C++11 standard, while version 1.8.x is the last releases that works with pre-C++11 compilers.
- For CentOS 7, the gcc (version 4.8.x) fully supports C++11, so the latest version would be prefered.
- For CentOS 6, the gcc (version 4.4.x) support only part of C++11, so only v1.8.x is available. Still, with the experimental C++0x, part of C++11 features is provided.
Get source code of googletest
You can clone the git repository, or download the source tarball.
In this example, release-v1.10.0 will be used. we'll build it under ~/build-googletest
Clone git repository
mkdir ~/build-googletestcd ~/build-googletestgit clone https://github.com/google/googletest.gitcd googletest# list all released versiongit tag# possible output: release-1.0.0 release-1.0.1 release-1.1.0 release-1.10.0 release-1.2.0 release-1.2.1 release-1.3.0 release-1.4.0 release-1.5.0 release-1.6.0 release-1.7.0 release-1.8.0 release-1.8.1 v1.10.x # unreleased# we'll check out the latest releasegit checkout release-1.10.0
Download source tarball
mkdir ~/build-googletestcd ~/build-googletestwget https://github.com/google/googletest/archive/release-1.10.0.tar.gztar xzf googletest-release-1.10.0.tar.gz
Build shared library
I've created a github repository alancprc/google-test-build-as-shared-lib, containing the Makefile to build googletest as a shared library, you should get the Makefile
before going on.
- Put the
Makefile
under~/build-googletest
- The default option in Makefile is
-m64
, which tells the compiler to generate code for a 64-bit environment.-m64
is for U1709+, you'll need to replace-m64
with-m32
for Unison 4.x - Run
make
to build If everything goes well, you'll get following .so files
libgtest.solibgtest_main.solibgmock.solibgmock_main.so
and the following output
Running main() from gmock_main.cc[==========] Running 6 tests from 2 test cases.[----------] Global test environment set-up.[----------] 3 tests from FactorialTest[ RUN ] FactorialTest.Negative[ OK ] FactorialTest.Negative (0 ms)[ RUN ] FactorialTest.Zero[ OK ] FactorialTest.Zero (0 ms)[ RUN ] FactorialTest.Positive[ OK ] FactorialTest.Positive (0 ms)[----------] 3 tests from FactorialTest (0 ms total)[----------] 3 tests from IsPrimeTest[ RUN ] IsPrimeTest.Negative[ OK ] IsPrimeTest.Negative (0 ms)[ RUN ] IsPrimeTest.Trivial[ OK ] IsPrimeTest.Trivial (0 ms)[ RUN ] IsPrimeTest.Positive[ OK ] IsPrimeTest.Positive (0 ms)[----------] 3 tests from IsPrimeTest (0 ms total)[----------] Global test environment tear-down[==========] 6 tests from 2 test cases ran. (0 ms total)[ PASSED ] 6 tests.
Install googletest to system directories
Put the generated shared libraries into
/usr/lib64
, or/usr/lib
if-m32
option is set. root permission is required.cp -at /usr/lib64/ libgtest*.so libgmock*.so
put the header files to /usr/local/include
cp -at /usr/local/include ./googletest/googletest/include/gtest ./googletest/googletest/include/gmock
Use googletest in Unison application library
Library to test
In this example, we create a Unison application library called UnisonUtil, containing files unison-util.h
and unison-util.cpp
, providing a simple function void removeTrailNewLine(StringS &str)
. The source code is as follows:
unison-util.h
#pragma once#include <Unison.h>namespace util {/** @brief remove trailing new line '\n' in string.*/void removeTrailNewLine(StringS &str);} // namespace util
unison-util.cpp
#include "unison-util.h"namespace util {void removeTrailNewLine(StringS &str){ int len = str.Length(); if (len and str[len - 1] == '\n') { str.Erase(len - 1, 1); }}} // namespace util
Create UnitTest application library
- Create an application library UnitTest in Unison TestTool.
Add a source file unittest.cpp, with following code
#include <gmock/gmock.h>#include "unison-util.h"using namespace std;TMResultM StartGTest(){ int argc = 1; const char* argv[] = {"a.out"}; std::cout << "\n\n\nRunning main() from unittest.cpp\n"; testing::InitGoogleMock(&argc, const_cast<char**>(argv)); int result = RUN_ALL_TESTS(); return result == 0 ? TM_PASS : TM_FAIL;}TEST(UtilFuncTest, removeTrailNewLineTest){ StringS str = "hello world\n"; util::removeTrailNewLine(str); util::removeTrailNewLine(str); EXPECT_STREQ(str, "hello world");}
Set up the UnitTest library
Add following to Compiler Flags
-pthread-DGTEST_LINKED_AS_SHARED_LIBRARY=1-fno-gnu-unique
Add following to Linker Flags
-lpthread-lgmock -lgtest
Add include path for UnisonUtil and googletest
./ # include path for UnisonUtil/usr/local/include # include path for googletest
- Add library UnisonUtil as dependency
Build and start test
- build UnitTest library
- Create a TestGroup
tgStartGTest
, which callsStartGTest()
function Execute the TestGroup
you should see the following output in dataviewerRunning main() from unittest.cpp[==========] Running 1 test from 1 test suite.[----------] Global test environment set-up.[----------] 1 test from UtilFuncTest[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[----------] 1 test from UtilFuncTest (0 ms total)[----------] Global test environment tear-down[==========] 1 test from 1 test suite ran. (0 ms total)[ PASSED ] 1 test.
Known issue
Unable to close the dynamic-linked library file
<< __WARNING 1:___________________________________In utlrt-aliang_sim (DynamicLink_Method), 4gl/class/dll_method/DLL_Installation.cxx, line 897, at Sun May 31 12:06:55 2020Warning: Uninstalling Dynamic-Link Method:File: /home/aliang/working/unison-util/x86_64_linux_3.10.0/libUnitTest.un.soUnable to close the dynamic-linked library file Please check for any static variables in inlined functions and move them out so they are not inlined. The test program has to be unloaded and loaded back after changing it. Running 'readelf -Ws libUnitTest.un.so | grep UNIQUE | c++filt' will help to find what is causing the error.User file: 4gl/class/dll_method/DLL_Installation.cxx, Line: 897, Statement: DynamicLink_Method
- This error may occur on CentOS 7 or CentOS 6 with an updated gcc version. To solve this error, add compiler flag
-fno-gnu-unique
to the application library.
- This error may occur on CentOS 7 or CentOS 6 with an updated gcc version. To solve this error, add compiler flag
Why not build googletest into a Unison application library?
Though installing googletest into system directory is disencouraged by googletest document, I found it necessary to use it in Unison.
I've tried to build googletest into a Unison application library, but when the library gets re-compiled, the tests get duplicated, which is really annoying.
For example, if the above test uses a googletest as a Unison application library, it'll become something like following after re-compiling UnisonUtil library.
Running main() from unittest.cpp[==========] Running 2 tests from 1 test suites.[----------] Global test environment set-up.[----------] 2 tests from UtilFuncTest[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[----------] 2 tests from UtilFuncTest (0 ms total)[----------] Global test environment tear-down[==========] 2 tests from 1 test suites ran. (0 ms total)[ PASSED ] 2 tests.
if the UnisonUtil library gets another re-compiling, the output becomes as follows.
Running main() from unittest.cpp[==========] Running 3 tests from 1 test suites.[----------] Global test environment set-up.[----------] 3 tests from UtilFuncTest[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[ RUN ] UtilFuncTest.removeTrailNewLineTest[ OK ] UtilFuncTest.removeTrailNewLineTest (0 ms)[----------] 3 tests from UtilFuncTest (0 ms total)[----------] Global test environment tear-down[==========] 3 tests from 1 test suites ran. (0 ms total)[ PASSED ] 3 tests.
原创文章,始发于https://alancprc.github.io/c++/2020/05/31/testing-in-unison-with-googletest.html
本文链接https://segmentfault.com/a/1190000022796424
转载请注明出处