mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-13 22:30:37 +08:00
commit
1323400996
@ -1,371 +0,0 @@
|
||||
[#]: subject: "Perform unit tests using GoogleTest and CTest"
|
||||
[#]: via: "https://opensource.com/article/22/1/unit-testing-googletest-ctest"
|
||||
[#]: author: "Stephan Avenwedde https://opensource.com/users/hansic99"
|
||||
[#]: collector: "lujun9972"
|
||||
[#]: translator: "toknow-gh"
|
||||
[#]: reviewer: " "
|
||||
[#]: publisher: " "
|
||||
[#]: url: " "
|
||||
|
||||
Perform unit tests using GoogleTest and CTest
|
||||
======
|
||||
Using unit tests will likely improve your code's quality and do so
|
||||
without disturbing your workflow.
|
||||
![Team checklist and to dos][1]
|
||||
|
||||
This article is a follow-up to my last article [Set up a build system with CMake and VSCodium][2].
|
||||
|
||||
In the last article, I showed how to configure a build system based on [VSCodium][3] and [CMake][4]. This article refines this setup by integrating meaningful unit tests using [GoogleTest][5] and [CTest][6].
|
||||
|
||||
If not already done, clone the [repository][7], open it in VSCodium and checkout the tag _devops_2_ by clicking on the _main_-branch symbol (red marker) and choosing the branch (yellow marker):
|
||||
|
||||
![VSCodium tag][8]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
Alternatively, open the command line and type:
|
||||
|
||||
|
||||
```
|
||||
`$ git checkout tags/devops_2`
|
||||
```
|
||||
|
||||
### GoogleTest
|
||||
|
||||
GoogleTest is a platform-independent, open source C++ testing framework. Even though GoogleTest is not meant to be exclusively for unit tests, I will use it to define unit tests for the _Generator_ library. In general, a unit test should verify the behavior of a single, logical unit. The _Generator_ library is one unit, so I'll write some meaningful tests to ensure proper function.
|
||||
|
||||
Using GoogleTest, the test cases are defined by assertions macros. Processing an assertion generates one of the following results:
|
||||
|
||||
* _Success_: Test passed.
|
||||
* _Nonfatal failure_: Test failed, but the test function will continue.
|
||||
* _Fatal failure_: Test failed, and the test function will be aborted.
|
||||
|
||||
|
||||
|
||||
The assertions macros follow this scheme to distinguish a fatal from a nonfatal failure:
|
||||
|
||||
* `ASSERT_*` fatal failure, function is aborted.
|
||||
* `EXPECT_*` nonfatal failure, function is not aborted.
|
||||
|
||||
|
||||
|
||||
Google recommends using `EXPECT_*` macros as they allow the test to continue when the tests define multiple assertions. An assertion macro takes two arguments: The first argument is the name of the test group (a freely selectable string), and the second argument is the name of the test itself. The _Generator_ library just defines the function _generate(...)_, therefore the tests in this article belong to the same group: _GeneratorTest_.
|
||||
|
||||
The following unit tests for the _generate(...)_ function can be found in [GeneratorTest.cpp][9].
|
||||
|
||||
#### Reference check
|
||||
|
||||
The [generate(...)][10] function takes a reference to a [std::stringstream][11] as an argument and returns the same reference. So the first test is to check if the passed reference is the same reference which the function returns.
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, ReferenceCheck){
|
||||
const int NumberOfElements = 10;
|
||||
std::stringstream buffer;
|
||||
EXPECT_EQ(
|
||||
std::addressof(buffer),
|
||||
std::addressof(Generator::generate(buffer, NumberOfElements))
|
||||
);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
Here I use [std::addressof][12] to check if the address of the returned object refers to the same object I provided as input.
|
||||
|
||||
#### Number of elements
|
||||
|
||||
This test checks if the number of elements in the stringstream reference matches the number given as an argument.
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, NumberOfElements){
|
||||
const int NumberOfElements = 50;
|
||||
int nCalcNoElements = 0;
|
||||
|
||||
std::stringstream buffer;
|
||||
|
||||
Generator::generate(buffer, NumberOfElements);
|
||||
std::string s_no;
|
||||
|
||||
while(std::getline(buffer, s_no, ' ')) {
|
||||
nCalcNoElements++;
|
||||
}
|
||||
|
||||
EXPECT_EQ(nCalcNoElements, NumberOfElements);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
#### Shuffle
|
||||
|
||||
This test checks the proper working of the random engine. If I invoke the _generate_ function two times in a row, I expect not to get the same result.
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, Shuffle){
|
||||
|
||||
const int NumberOfElements = 50;
|
||||
|
||||
std::stringstream buffer_A;
|
||||
std::stringstream buffer_B;
|
||||
|
||||
Generator::generate(buffer_A, NumberOfElements);
|
||||
Generator::generate(buffer_B, NumberOfElements);
|
||||
|
||||
EXPECT_NE(buffer_A.str(), buffer_B.str());
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
#### Checksum
|
||||
|
||||
This is the largest test. It checks whether the sum of the digits of a numerical series from 1 to _n_ is the same as the sum of the shuffled output series. I expect that the sum matches as the _generate(...)_ function should simply create a shuffled variant of such a series.
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, CheckSum){
|
||||
|
||||
const int NumberOfElements = 50;
|
||||
int nChecksum_in = 0;
|
||||
int nChecksum_out = 0;
|
||||
|
||||
std::vector<int> vNumbersRef(NumberOfElements); // Input vector
|
||||
std::iota(vNumbersRef.begin(), vNumbersRef.end(), 1); // Populate vector
|
||||
|
||||
// Calculate reference checksum
|
||||
for(const int n : vNumbersRef){
|
||||
nChecksum_in += n;
|
||||
}
|
||||
|
||||
std::stringstream buffer;
|
||||
Generator::generate(buffer, NumberOfElements);
|
||||
|
||||
std::vector<int> vNumbersGen; // Output vector
|
||||
std::string s_no;
|
||||
|
||||
// Read the buffer back back to the output vector
|
||||
while(std::getline(buffer, s_no, ' ')) {
|
||||
vNumbersGen.push_back(std::stoi(s_no));
|
||||
}
|
||||
|
||||
// Calculate output checksum
|
||||
for(const int n : vNumbersGen){
|
||||
nChecksum_out += n;
|
||||
}
|
||||
|
||||
EXPECT_EQ(nChecksum_in, nChecksum_out);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
The above tests can also be debugged like an ordinary C++ application.
|
||||
|
||||
### CTest
|
||||
|
||||
In addition to the in-code unit test, the [CTest][6] utility lets me define tests that can be performed on executables. In a nutshell, I call the executable with certain arguments and match the output with [regular expressions][13]. This lets me simply check how the executable behaves with incorrect command-line arguments. The tests are defined in the top level [CMakeLists.txt][14]. Here is a closer look at three test cases:
|
||||
|
||||
#### Regular usage
|
||||
|
||||
If a positive integer is provided as a command-line argument, I expect the executable to produce a series of numbers separated by whitespace:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME RegularUsage COMMAND Producer 10)
|
||||
set_tests_properties(RegularUsage
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^[0-9 ]+"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### No argument
|
||||
|
||||
If no argument is provided, the program should exit immediately and display the reason why:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME NoArg COMMAND Producer)
|
||||
set_tests_properties(NoArg
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^Enter the number of elements as argument"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### Wrong argument
|
||||
|
||||
Providing an argument that cannot be converted into an integer should also cause an immediate exit with an error message. This test invokes the _Producer_ executable with the command line parameter*"ABC"*:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME WrongArg COMMAND Producer ABC)
|
||||
set_tests_properties(WrongArg
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^Error: Cannot parse"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### Testing the tests
|
||||
|
||||
To run a single test and see how it is processed, invoke `ctest` from the command line providing the following arguments:
|
||||
|
||||
* Run single tst: `-R <test-name>`
|
||||
* Enable verbose output: `-VV`
|
||||
|
||||
|
||||
|
||||
Here is the command `ctest -R Usage -VV:`
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
$ ctest -R Usage -VV
|
||||
UpdatecTest Configuration from :/home/stephan/Documents/cpp_testing sample/build/DartConfiguration.tcl
|
||||
UpdateCTestConfiguration from :/home/stephan/Documents/cpp_testing sample/build/DartConfiguration.tcl
|
||||
Test project /home/stephan/Documents/cpp_testing sample/build
|
||||
Constructing a list of tests
|
||||
Done constructing a list of tests
|
||||
Updating test list for fixtures
|
||||
Added 0 tests to meet fixture requirements
|
||||
Checking test dependency graph...
|
||||
Checking test dependency graph end
|
||||
|
||||
```
|
||||
|
||||
In this code block, I invoked a test named _Usage_.
|
||||
|
||||
This ran the executable with no command-line arguments:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
test 3
|
||||
Start 3: Usage
|
||||
3: Test command: /home/stephan/Documents/cpp testing sample/build/Producer
|
||||
|
||||
```
|
||||
|
||||
The test failed because the output didn't match the regular expression `[^[0-9]+]`.
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
3: Enter the number of elements as argument
|
||||
1/1 test #3. Usage ................
|
||||
|
||||
Failed Required regular expression not found.
|
||||
Regex=[^[0-9]+]
|
||||
|
||||
0.00 sec round.
|
||||
|
||||
0% tests passed, 1 tests failed out of 1
|
||||
Total Test time (real) =
|
||||
0.00 sec
|
||||
The following tests FAILED:
|
||||
3 - Usage (Failed)
|
||||
Errors while running CTest
|
||||
$
|
||||
|
||||
```
|
||||
|
||||
To run all tests (including the one defined with GoogleTest), navigate to the _build_ directory and run `ctest`:
|
||||
|
||||
![CTest run][15]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
Inside VSCodium, click on the area marked yellow in the info bar to invoke CTest. If all tests pass, the following output is displayed:
|
||||
|
||||
![VSCodium][16]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
### Automate testing with Git Hooks
|
||||
|
||||
By now, running the tests is an additional step for the developer. The developer could also commit and push code that doesn't pass the tests. Thanks to [Git Hooks][17], I can implement a mechanism that automatically runs the tests and prevents the developer from accidentally committing faulty code.
|
||||
|
||||
Navigate to `.git/hooks`, create an empty file named _pre-commit_, and copy and paste the following code:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
#!/usr/bin/sh
|
||||
|
||||
(cd build; ctest --output-on-failure -j6)
|
||||
|
||||
```
|
||||
|
||||
After it, make this file executable:
|
||||
|
||||
|
||||
```
|
||||
`$ chmod +x pre-commit`
|
||||
```
|
||||
|
||||
This script invokes CTest when trying to perform a commit. If a test fails, like in the screenshot below, the commit is aborted:
|
||||
|
||||
![Commit failed][18]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
If the tests succeed, the commit is processed, and the output looks like this:
|
||||
|
||||
![Commit succeeded][19]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
The described mechanism is only a soft barrier: A developer could still commit faulty code using `git commit --no-verify`. I can ensure that only working code is pushed by configuring a build server. This topic will be part of a separate article.
|
||||
|
||||
### Summary
|
||||
|
||||
The techniques mentioned in this article are easy to implement and help you quickly find bugs in your code. Using unit tests will likely improve your code's quality and, as I have shown, do so without disturbing your workflow. The GoogleTest framework provides features for every conceivable scenario; I only used a subset of its functionality. At this point, I also want to mention the [GoogleTest Primer][20], which gives you an overview of the ideas, opportunities, and features of the framework.
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
via: https://opensource.com/article/22/1/unit-testing-googletest-ctest
|
||||
|
||||
作者:[Stephan Avenwedde][a]
|
||||
选题:[lujun9972][b]
|
||||
译者:[译者ID](https://github.com/译者ID)
|
||||
校对:[校对者ID](https://github.com/校对者ID)
|
||||
|
||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||
|
||||
[a]: https://opensource.com/users/hansic99
|
||||
[b]: https://github.com/lujun9972
|
||||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/todo_checklist_team_metrics_report.png?itok=oB5uQbzf (Team checklist and to dos)
|
||||
[2]: https://opensource.com/article/22/1/devops-cmake
|
||||
[3]: https://vscodium.com/
|
||||
[4]: https://cmake.org/
|
||||
[5]: https://github.com/google/googletest
|
||||
[6]: https://cmake.org/cmake/help/latest/manual/ctest.1.html
|
||||
[7]: https://github.com/hANSIc99/cpp_testing_sample
|
||||
[8]: https://opensource.com/sites/default/files/cpp_unit_test_vscodium_tag.png (VSCodium tag)
|
||||
[9]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/Generator/GeneratorTest.cpp
|
||||
[10]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/Generator/Generator.cpp
|
||||
[11]: https://en.cppreference.com/w/cpp/io/basic_stringstream
|
||||
[12]: https://en.cppreference.com/w/cpp/memory/addressof
|
||||
[13]: https://en.wikipedia.org/wiki/Regular_expression
|
||||
[14]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/CMakeLists.txt
|
||||
[15]: https://opensource.com/sites/default/files/cpp_unit_test_ctest_run.png (CTest run)
|
||||
[16]: https://opensource.com/sites/default/files/cpp_unit_test_ctest_vscodium.png (VSCodium)
|
||||
[17]: https://git-scm.com/book/en/v2/Customizing-Git-Git-Hooks
|
||||
[18]: https://opensource.com/sites/default/files/cpp_unit_test_git_hook_commit_failed.png (Commit failed)
|
||||
[19]: https://opensource.com/sites/default/files/cpp_unit_test_git_hook_commit_succeeded.png (Commit succeeded)
|
||||
[20]: https://google.github.io/googletest/primer.html
|
@ -0,0 +1,369 @@
|
||||
[#]: subject: "Perform unit tests using GoogleTest and CTest"
|
||||
[#]: via: "https://opensource.com/article/22/1/unit-testing-googletest-ctest"
|
||||
[#]: author: "Stephan Avenwedde https://opensource.com/users/hansic99"
|
||||
[#]: collector: "lujun9972"
|
||||
[#]: translator: "toknow-gh"
|
||||
[#]: reviewer: " "
|
||||
[#]: publisher: " "
|
||||
[#]: url: " "
|
||||
|
||||
使用 GoogleTest 和 CTest 进行单元测试
|
||||
======
|
||||
进行单元测试可以提高代码质量,并且它不会打断你的工作流。
|
||||
![Team checklist and to dos][1]
|
||||
|
||||
|
||||
本文是 [使用 CMake 和 VSCodium 搭建构建系统][2] 的后续文章。
|
||||
|
||||
在上一篇文章中我介绍了基于 [VSCodium][3] 和 [CMake][4] 配置构建系统。本文我将介绍如何通过 [GoogleTest][5] 和 [CTest][6] 将单元测试集成到这个构建系统中。
|
||||
|
||||
首先克隆 [这个仓库][7],用 VSCodium 打开,切换到 `devops_2` 标签。你可以通过点击 `main` 分支(红框处),然后选择 `devops_2` 标签(黄框处)来进行切换:
|
||||
|
||||
![VSCodium tag][8]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
或者你可以通过命令行来切换:
|
||||
|
||||
|
||||
```
|
||||
`$ git checkout tags/devops_2`
|
||||
```
|
||||
|
||||
### GoogleTest
|
||||
|
||||
GoogleTest 是一个平台无关的开源 C++ 测试框架。单元测试是用来验证单个逻辑单元的行为的。尽管 GoogleTest 并不是专门用于单元测试的,我将用它对 `Generator` 库进行单元测试。
|
||||
|
||||
在 GoogleTest 中,测试用例是通过断言宏来定义的。断言可能产生以下结果:
|
||||
|
||||
* _成功_: 测试通过。
|
||||
* _非致命失败_: 测试失败,但测试继续。
|
||||
* _致命失败_: 测试失败,且测试终止。
|
||||
|
||||
|
||||
致命断言和非致命断言通过不同的宏来区分:
|
||||
|
||||
* `ASSERT_*`: 致命断言,失败时终止。
|
||||
* `EXPECT_*`: 非致命断言,失败时不终止。
|
||||
|
||||
|
||||
谷歌推荐使用 `EXPECT_*` 宏,因为当测试中包含多个的断言时,它允许继续执行。断言有两个参数:第一个参数是测试分组的名称,第二个参数是测试自己的名称。`Generator` 只定义了 `generate(...)` 函数,所以本文中所有的测试都属于同一个测试组:`GeneratorTest`。
|
||||
|
||||
针对 `generate(...)` 函数的测试可以从 [GeneratorTest.cpp][9] 中找到。
|
||||
|
||||
#### 引用一致性检查
|
||||
|
||||
[generate(...)][10] 函数有一个 [std::stringstream][11] 的引用作为输入参数,并且它也将这个引用作为返回值。第一个测试就是检查输入的引用和返回的引用是否一致。
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, ReferenceCheck){
|
||||
const int NumberOfElements = 10;
|
||||
std::stringstream buffer;
|
||||
EXPECT_EQ(
|
||||
std::addressof(buffer),
|
||||
std::addressof(Generator::generate(buffer, NumberOfElements))
|
||||
);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
在这个测试中我使用 [std::addressof][12] 来获取对象的地址,并用 `EXPECT_EQ` 来比较输入对象和返回对象是否是同一个。
|
||||
|
||||
#### 检查元素个数
|
||||
|
||||
本测试检查作为输入的 `std::stringstream` 引用中的元素个数与输入参数中指定的个数是否相同。
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, NumberOfElements){
|
||||
const int NumberOfElements = 50;
|
||||
int nCalcNoElements = 0;
|
||||
|
||||
std::stringstream buffer;
|
||||
|
||||
Generator::generate(buffer, NumberOfElements);
|
||||
std::string s_no;
|
||||
|
||||
while(std::getline(buffer, s_no, ' ')) {
|
||||
nCalcNoElements++;
|
||||
}
|
||||
|
||||
EXPECT_EQ(nCalcNoElements, NumberOfElements);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
#### 乱序重排
|
||||
|
||||
本测试检查随机化引擎是否工作正常。如果连续调用两次 `generate` 函数,应该得到的是两个不同的结果。
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, Shuffle){
|
||||
|
||||
const int NumberOfElements = 50;
|
||||
|
||||
std::stringstream buffer_A;
|
||||
std::stringstream buffer_B;
|
||||
|
||||
Generator::generate(buffer_A, NumberOfElements);
|
||||
Generator::generate(buffer_B, NumberOfElements);
|
||||
|
||||
EXPECT_NE(buffer_A.str(), buffer_B.str());
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
#### 求和校验
|
||||
|
||||
与前面的测试相比,这是一个大体量的测试。它检查 1 到 n 的数值序列的和与乱序重排后的序列的和是否相等。 `generate(...)` 函数应该生成一个 1 到 n 的乱序的序列,这个序列的和应当是不变的。
|
||||
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
TEST(GeneratorTest, CheckSum){
|
||||
|
||||
const int NumberOfElements = 50;
|
||||
int nChecksum_in = 0;
|
||||
int nChecksum_out = 0;
|
||||
|
||||
std::vector<int> vNumbersRef(NumberOfElements); // Input vector
|
||||
std::iota(vNumbersRef.begin(), vNumbersRef.end(), 1); // Populate vector
|
||||
|
||||
// Calculate reference checksum
|
||||
for(const int n : vNumbersRef){
|
||||
nChecksum_in += n;
|
||||
}
|
||||
|
||||
std::stringstream buffer;
|
||||
Generator::generate(buffer, NumberOfElements);
|
||||
|
||||
std::vector<int> vNumbersGen; // Output vector
|
||||
std::string s_no;
|
||||
|
||||
// Read the buffer back back to the output vector
|
||||
while(std::getline(buffer, s_no, ' ')) {
|
||||
vNumbersGen.push_back(std::stoi(s_no));
|
||||
}
|
||||
|
||||
// Calculate output checksum
|
||||
for(const int n : vNumbersGen){
|
||||
nChecksum_out += n;
|
||||
}
|
||||
|
||||
EXPECT_EQ(nChecksum_in, nChecksum_out);
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
你可以像对一般 C++ 程序一样调试这些测试。
|
||||
|
||||
### CTest
|
||||
|
||||
除了嵌入到代码中的测试之外,[CTest][6] 提供了可执行程序的测试方式。简而言之就是通过给可执行程序传入特定的参数,然后用 [正则表达式][13] 对它的输出进行匹配检查。通过这种方式可以很容易检查程序对于不正确的命令行参数的反应。这些测试定义在顶层的 [CMakeLists.txt][14] 文件中。下面我详细介绍 3 个测试用例:
|
||||
|
||||
#### 参数正常
|
||||
|
||||
如果输入参数是一个正整数,程序应该输出应该是一个数列:
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME RegularUsage COMMAND Producer 10)
|
||||
set_tests_properties(RegularUsage
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^[0-9 ]+"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### 没有提供参数
|
||||
|
||||
如果没有传入参数,程序应该立即退出并提示错误原因:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME NoArg COMMAND Producer)
|
||||
set_tests_properties(NoArg
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^Enter the number of elements as argument"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### 参数错误
|
||||
|
||||
当传入的参数不是整数时,程序应该退出并报错。比如给 `Producer` 传入参数 `ABC`:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
add_test(NAME WrongArg COMMAND Producer ABC)
|
||||
set_tests_properties(WrongArg
|
||||
PROPERTIES PASS_REGULAR_EXPRESSION "^Error: Cannot parse"
|
||||
)
|
||||
|
||||
```
|
||||
|
||||
#### 执行测试
|
||||
|
||||
可以使用 `ctest -R Usage -VV` 命令来执行测试。这里给 `ctest` 的命令行参数:
|
||||
|
||||
* `-R <测试名称>` : 执行单个测试
|
||||
* `-VV`:打印详细输出
|
||||
|
||||
|
||||
|
||||
测试执行结果如下:
|
||||
|
||||
```
|
||||
|
||||
|
||||
$ ctest -R Usage -VV
|
||||
UpdatecTest Configuration from :/home/stephan/Documents/cpp_testing sample/build/DartConfiguration.tcl
|
||||
UpdateCTestConfiguration from :/home/stephan/Documents/cpp_testing sample/build/DartConfiguration.tcl
|
||||
Test project /home/stephan/Documents/cpp_testing sample/build
|
||||
Constructing a list of tests
|
||||
Done constructing a list of tests
|
||||
Updating test list for fixtures
|
||||
Added 0 tests to meet fixture requirements
|
||||
Checking test dependency graph...
|
||||
Checking test dependency graph end
|
||||
|
||||
```
|
||||
|
||||
在这里我执行了名为 `Usage` 的测试。
|
||||
|
||||
它以无参数的方式调用 `Producer`:
|
||||
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
test 3
|
||||
Start 3: Usage
|
||||
3: Test command: /home/stephan/Documents/cpp testing sample/build/Producer
|
||||
|
||||
```
|
||||
|
||||
输出不匹配 `[^[0-9]+]` 的正则模式,测试未通过。
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
3: Enter the number of elements as argument
|
||||
1/1 test #3. Usage ................
|
||||
|
||||
Failed Required regular expression not found.
|
||||
Regex=[^[0-9]+]
|
||||
|
||||
0.00 sec round.
|
||||
|
||||
0% tests passed, 1 tests failed out of 1
|
||||
Total Test time (real) =
|
||||
0.00 sec
|
||||
The following tests FAILED:
|
||||
3 - Usage (Failed)
|
||||
Errors while running CTest
|
||||
$
|
||||
|
||||
```
|
||||
|
||||
如果想要执行所有测试(包括那些用 GoogleTest 生成的),切换到 `build` 目录中,然后运行 `ctest` 即可:
|
||||
|
||||
![CTest run][15]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
在 VSCodium 中可以通过点击信息栏的黄框处来调用 CTest。如果所有测试都通过了,你会看到如下输出:
|
||||
|
||||
![VSCodium][16]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
### 使用 Git 钩子进行自动化测试
|
||||
|
||||
目前为止,运行测试是开发者需要额外执行的步骤,那些不能通过测试的代码仍然可能被提交和推送到代码仓库中。利用 [Git 钩子][17] 可以自动执行测试,从而防止有瑕疵的代码被提交。
|
||||
|
||||
切换到 `.git/hooks` 目录,创建 `pre-commit` 文件,复制粘贴下面的代码:
|
||||
|
||||
|
||||
```
|
||||
|
||||
|
||||
#!/usr/bin/sh
|
||||
|
||||
(cd build; ctest --output-on-failure -j6)
|
||||
|
||||
```
|
||||
|
||||
然后,给文件增加可执行权限:
|
||||
|
||||
```
|
||||
`$ chmod +x pre-commit`
|
||||
```
|
||||
|
||||
这个脚本会在提交之前调用 CTest 进行测试。如果有测试未通过,提交过程就会被终止:
|
||||
|
||||
![Commit failed][18]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
只有所有测试都通过了,提交过程才会完成:
|
||||
|
||||
![Commit succeeded][19]
|
||||
|
||||
Stephan Avenwedde (CC BY-SA 4.0)
|
||||
|
||||
这个机制也有一个漏洞:可以通过 `git commit --no-verify` 命令绕过测试。解决办法是配置构建服务器,这能保证只有正常工作的代码才能被提交,但这又是另一个话题了。
|
||||
|
||||
### 总结
|
||||
|
||||
本文提到的技术实施简单,并且能够帮你快速发现代码中的 bug。做单元测试可以提高代码质量,同时也不会打断你的工作流。GoogleTest 框架提供了丰富的特性以应对各种测试场景,文中我所提到的只是一小部分而已。如果你想进一步了解 GoogleTest,我推荐你阅读 [GoogleTest Primer][20]。
|
||||
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
via: https://opensource.com/article/22/1/unit-testing-googletest-ctest
|
||||
|
||||
作者:[Stephan Avenwedde][a]
|
||||
选题:[lujun9972][b]
|
||||
译者:[toknow-gh](https://github.com/toknow-gh)
|
||||
校对:[校对者ID](https://github.com/校对者ID)
|
||||
|
||||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||||
|
||||
[a]: https://opensource.com/users/hansic99
|
||||
[b]: https://github.com/lujun9972
|
||||
[1]: https://opensource.com/sites/default/files/styles/image-full-size/public/lead-images/todo_checklist_team_metrics_report.png?itok=oB5uQbzf (Team checklist and to dos)
|
||||
[2]: https://opensource.com/article/22/1/devops-cmake
|
||||
[3]: https://vscodium.com/
|
||||
[4]: https://cmake.org/
|
||||
[5]: https://github.com/google/googletest
|
||||
[6]: https://cmake.org/cmake/help/latest/manual/ctest.1.html
|
||||
[7]: https://github.com/hANSIc99/cpp_testing_sample
|
||||
[8]: https://opensource.com/sites/default/files/cpp_unit_test_vscodium_tag.png (VSCodium tag)
|
||||
[9]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/Generator/GeneratorTest.cpp
|
||||
[10]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/Generator/Generator.cpp
|
||||
[11]: https://en.cppreference.com/w/cpp/io/basic_stringstream
|
||||
[12]: https://en.cppreference.com/w/cpp/memory/addressof
|
||||
[13]: https://en.wikipedia.org/wiki/Regular_expression
|
||||
[14]: https://github.com/hANSIc99/cpp_testing_sample/blob/main/CMakeLists.txt
|
||||
[15]: https://opensource.com/sites/default/files/cpp_unit_test_ctest_run.png (CTest run)
|
||||
[16]: https://opensource.com/sites/default/files/cpp_unit_test_ctest_vscodium.png (VSCodium)
|
||||
[17]: https://git-scm.com/book/en/v2/Customizing-Git-Git-Hooks
|
||||
[18]: https://opensource.com/sites/default/files/cpp_unit_test_git_hook_commit_failed.png (Commit failed)
|
||||
[19]: https://opensource.com/sites/default/files/cpp_unit_test_git_hook_commit_succeeded.png (Commit succeeded)
|
||||
[20]: https://google.github.io/googletest/primer.html
|
Loading…
Reference in New Issue
Block a user