google test framework (public)

40
An Introduction to Google Test Framework Abner Huang July. 2015

Upload: abner-huang

Post on 19-May-2015

394 views

Category:

Engineering


5 download

TRANSCRIPT

Page 1: Google test framework (public)

An Introduction to Google Test Framework

Abner HuangJuly. 2015

Page 2: Google test framework (public)

Test!? We do it every day

Page 3: Google test framework (public)

void Test__calc_isect() {

int x1, y1, x2, y2;

int x3, y3, x4, y4;

int x, y;

x1 = 0; y1 = INT_MIN; x2 = 2; y2 = INT_MAX;

x3 = 20; y3 = INT_MAX; x4 = 22; y4 = INT_MIN;

calc_isect(&x, &y, x1, y1, x2, y2, x3, y3, x4, y4, 0);

assert(x == 11);

assert(y == INT_MAX);

}

Page 4: Google test framework (public)

void main(void)

{

Test__calc_isect1();

Test__calc_isect2();

Test__calc_isect3();

Test__calc_isect();

Test__calc_isect4();

Test__calc_isect5();

Test__calc_isect6();

}

Page 5: Google test framework (public)

> ./test.out

test.cpp:100:Test__calc_isect(): Assertion `x == 11' failed.

Page 6: Google test framework (public)

void Test__calc_isect() {

int x1, y1, x2, y2;

int x3, y3, x4, y4;

int x, y;

x1 = 0; y1 = INT_MIN; x2 = 2; y2 = INT_MAX;

x3 = 20; y3 = INT_MAX; x4 = 22; y4 = INT_MIN;

calc_isect(&x, &y, x1, y1, x2, y2, x3, y3, x4, y4, 0);

if ((x != 11) || (y != INT_MAX))

{

printf(“Oops! (%d,%d) != (%d,%d)\n”,x,y,11,INT_MAX);

assert(0);

}

printf(“PASS\n”);

}

Page 7: Google test framework (public)

> ./test.out

PASS

PASS

PASSOops! (10,2147483647) != (11,2147483647)

test.cpp:100:Test__calc_isect(): Assertion `0' failed.

Page 8: Google test framework (public)

Any complains?

Page 9: Google test framework (public)

No convenient controls

Page 10: Google test framework (public)

Run in batch mode

Page 11: Google test framework (public)

Does gtest help?

Page 12: Google test framework (public)

No control of test suite and test cases

Page 13: Google test framework (public)

./your_test.out --help

● Selecting Tests

● Repeating/Shuffling the Tests

● Generating an XML Report

● Turning Assertion Failures into Break-Points○ --gtest_break_on_failure

● Disabling Catching Test-Thrown Exceptions○ --gtest_catch_exceptions=0

Page 14: Google test framework (public)

> libpl_test --gtest_list_tests

CHK_ENCLOSE.

PlEnclose_SimpleInput

MAKE_BD.

MakeBd_LineTouchDirty

OSIZE_HDR.

OversizeHdr_BadOversizAmount

OversizeHdr_NULLInput

OversizeHdr_SimpleInput

OversizeHdr_OversizeOrthoHdr

OversizeHdr_OversizeNonOrthoHdr

OversizeOrthoHdr_OversizeOrthoHdr

BreakUpAngleHdr_DisconnectedInput

BreakUpAngleHdr_AngleBigInput

Page 15: Google test framework (public)

https://www.flickr.com/photos/marckjerland/4285433177 On Flickr, marc kjerland's Mathematician's toolkit

Page 16: Google test framework (public)

● Fatal and non-fatal assertions○ EXPECT_EQ(x, 10);EXPECT_FLOAT_EQ(x,10.0)

○ ASSERT_EQ(x, 10);ASSERT_DOUBLE_EQ(x,10.0)

● Easy assertions informative messages:assert(x == 10);

EXPECT_EQ(x, 10) << “vector (” << old_x <<

”,” << old_y << ”) moved to (” << x << ”,” <<

y << ”)\n”;

Page 17: Google test framework (public)

> ./test.out

[ RUN ] MYTEST.TestFoo

test.cpp:37: Failure

Value of: x

Actual: 11

Expected: 10

vector (10, 77) move to (11, 77)

[ FAILED ] MYTEST.TestFoo (0 ms)

Page 18: Google test framework (public)

DEATH TEST calc_isect(int *x, int *y, int x1, int y1, int x2, int y2,

int x3, int y3, int x4, int y4, int tolerance)

{

if (x==NULL || y==NULL)

{

cerr << “Input is invalid.\n”;

exit(1);

}

… …}

TEST(GEODeathTest, CalcIsect) {

ASSERT_DEATH(CalcIsect(NULL, NULL, 0,0,0,0,0,0,0,0,0),

".*invalid");

}

Page 19: Google test framework (public)

Adding Traces to Assertions

void Sub1(int n) {

EXPECT_EQ(1, Bar(n));

EXPECT_EQ(2, Bar(n + 1));

}

TEST(FooTest, Bar) {

{

SCOPED_TRACE("A");

Sub1(1);

}

Sub1(9);

}

path/to/foo_test.cc:11: Failure

Value of: Bar(n)

Expected: 1

Actual: 2

Trace:

path/to/foo_test.cc:17: A

path/to/foo_test.cc:12: Failure

Value of: Bar(n + 1)

Expected: 2

Page 20: Google test framework (public)

XML outputs<testsuites name="AllTests" tests="3" failures="

1" errors="0" time="35" ...>

<testsuite name="test_case_name" tests="2"

failures="1" errors="0" time="15"...>

<testcase name="test_name" status="run"

time="7"...>

<failure message="..."/>

<failure message="..."/>

</testcase>

</testsuite>

</testsuites>

Page 21: Google test framework (public)

Prepare your test case• test.cpp contains main() in test directory

#include "_test_util.h"

static void

infrastructure_init(void)

{

vmemUseMalloc(1);

vmemInit_m();

vmemSetIncrementValue(VMEM_MEGABYTE_MINUS);

hoVMsetDefaultPoolLockingState(0);

vth_init();

}

int main(int argc, char **argv) {

infrastructure_init(); // Write a function like this to initialize vmem, threading etc.

::testing::InitGoogleTest(&argc, argv);

int result = RUN_ALL_TESTS(); Initialize gtest and runs all tests.

return result;

}

Page 22: Google test framework (public)

SetUp()

TEST/TEST_F/TEST_PTearDown()

Test types of Gtest

• TEST()• TEST_F() -- Fixture

• TEST_P() -- Value-parameterized fixture– Range()– Values()– ValuesIn()– Bool()– Combine()

Page 23: Google test framework (public)

TEST(GEO, CalcIsect) {

int x1, y1, x2, y2;

int x3, y3, x4, y4;

int x, y;

x1 = 0; y1 = INT_MIN; x2 = 2; y2 = INT_MAX;

x3 = 20; y3 = INT_MAX; x4 = 22; y4 = INT_MIN;

calc_isect(&x, &y, x1, y1, x2, y2, x3, y3, x4, y4, 0);

assert(x == 11);

assert(y == INT_MAX);

}

Page 24: Google test framework (public)

Test Fixtures

• Sometimes you want to run several tests on similar data structures (same setup before running test).

• In a fixture you define common setup/tear-down for multiple tests

• Reduces amount of test code by avoiding repetition

Page 25: Google test framework (public)

class GEO : public ::testing::Test {

protected:

virtual void SetUp() {}

void ZapMemoryPool() {}

virtual void TearDown() {}

int x1, y1, x2, y2, x3, y3, x4, y4, x, y;

};

TEST_F(GEO, CalcIsect) {

x1 = 0; y1 = INT_MIN; x2 = 2; y2 = INT_MAX;

x3 = 20; y3 = INT_MAX; x4 = 22; y4 = INT_MIN;

calc_isect(&x, &y, x1, y1, x2, y2, x3, y3, x4, y4, 0);

assert(x == 11);

assert(y == INT_MAX);

}

Page 26: Google test framework (public)

TEST_P() -- Value-parameterized fixture

● generator based tests.

Range(begin, end[, step])

Yields values {begin, begin+step, begin+step+step, ...}. The values do not include end. step defaults to 1.

Values(v1, v2, ..., vN) Yields values {v1, v2, ..., vN}.

ValuesIn(container) and ValuesIn(begin, end)

Yields values from a C-style array, an STL-style container, or an iterator range [begin, end). container, begin, and end can be expressions whose values are determined at run time.

Bool() Yields sequence {false, true}.

Combine(g1, g2, ..., gN)

Yields all combinations (the Cartesian product for the math savvy) of the values generated by the N generators. This is only available if your system provides the <tr1/tuple> header.

Page 27: Google test framework (public)

class GEO : public ::testing::TestWithParam<int> {

protected:

virtual void SetUp() {}

void ZapMemoryPool() {}

virtual void TearDown() {}

int x1, y1, x2, y2, x3, y3, x4, y4, x, y;

};

INSTANTIATE_TEST_CASE_P(AUTO_GEN_OH, GEO, ::testing::Range(1,12,1));

TEST_P(GEO, CalcIsect) {

double angle = PI/180 * 15 * GetParam();

x1 = -100; y1 = 0; x2 = 100; y2 = 0;

x3 = -50; y3 = 0; x4 = 50; y4 = 0;

rotate(&x3, &y4, angle);

rotate(&x4, &y4, angle);

calc_isect(&x, &y, x1, y1, x2, y2, x3, y3, x4, y4, 0);

assert(x == 0 && y == 0);

}

Page 28: Google test framework (public)

[----------] 16 tests from AUTO_GEN_OH/GEO

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/1

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/1 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/2

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/2 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/3

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/3 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/4

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/4 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/5

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/5 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/6

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/6 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/7

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/7 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/8

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/8 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/9

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/9 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/10

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/10 (0 ms)

[ RUN ] AUTO_GEN_OH/GEO.CalcIsect/11

[ OK ] AUTO_GEN_OH/GEO.CalcIsect/11 (0 ms)

[----------] 11 tests from AUTO_GEN_OH/GEO (0 ms total)

Page 29: Google test framework (public)

Organization of Tests

TEST(GEO, CalcIsect)test case test

Test Suite

Page 30: Google test framework (public)

Flow of Running Tests

SetUp()

TEST/TEST_F/TEST_PTearDown()

Environment::SetUp()

SetUpTestCase()

Environment::TearDown()

TearDownTestCase()

Page 31: Google test framework (public)

How to install gtest

Page 32: Google test framework (public)

Easy Install> wget http://googletest.googlecode.com/files/gtest-1.7.0.zip

> unzip gtest-1.7.0.zip

> cd gtest-1.7.0

> ./configure

> make

> cp -a include/gtest /usr/include

> sudo cp -a lib/.libs/* /usr/lib/

Page 33: Google test framework (public)

How to use google test

• You must check out gtest to use unit test feature. We do not have the executable of gtest.

• E.g.– /path/to/foo/

– build

– arch

– src

– exec

– gtest

– include

– lib

Page 34: Google test framework (public)

How to use google test (II)

• E.g./path/to/foo/

– exec

– gtest

– include

– lib

– bar <<< Library under test, contains production code

– published

– test <<< Directory contains test code

test> make

Page 35: Google test framework (public)

Some points to keep in mind

• gtest is useful for testing primitive functions in libraries• Unit testing does not replace regression testing• With gtest:

– You do not modify production code. – You add test code that calls library functions.– You check-in test code to repository

• Production code is not compiled/linked with gtest• Using gtest creates a test binary, not icv binary

Page 36: Google test framework (public)

Coding Style and Rules

• Don’t mix TEST(), TEST_F(), TEST_P()• Don’t use global variables, use test fixture to instead• Gtest does not reuse the same test fixture object across

multiple tests• Spell SetUp() correctly.• Gtest use fork() to do death test, do not try not to free

memory in a death test.• You can derive new class from your own class.

Page 37: Google test framework (public)

Coding Style and Rules

• Use Camel Case style to name test cases and suite– E.g., TEST(Time, Flies_Like_An_Arrow) { ... }

and TEST(Time_Flies, Like_An_Arrow) { … }will have the same name Time_Files_Like_An_Arrow_Test

• Current Naming Rule– Test suite name is all uppercase– Test case name is Camel Case style with underscore.

– E.g., TEST_F(SPACING_UTIL__PNT_EDGE, WhichHalfRes_SamePoints)

Page 38: Google test framework (public)

Key Features & Characters Boost Test Google Test UnitTest++

Free (Open Source) software Yes Yes Yes

Simple to use Yes(5 out of 5)

Yes(5 out of 5)

Yes(4 out of 5)

Test cases grouping & organization Good Good Normal

Test case template support Yes Yes Yes

Test fixture (i.e. fixed state used as a baseline for running test) support

Yes Yes Yes

Global test fixture Yes Yes No

Handle C++ exceptions Yes Yes Yes

Different types of assertions support Yes Yes No

Fatal and non-fatal assertions Yes Yes No

Support Regular Expression when matching function output

Yes Yes No

Output in XML format Yes Yes No

Page 39: Google test framework (public)

Key Features & Characters Boost Test Google Test UnitTest++

Support Mock object (objects can simulate the behavior of complex objects)

No Yes(work seamlessly with

Google Mock)

No

Extend features with event mechanism

No Yes No

Run tests on distributed machines No Yes No

Thread safety when running tests No Yes No

Run subset of all defined test cases Yes(only support

filtering by test name)

Yes(support multiple

filtering conditions)

No

• Karl Chen, C++ Unit Testing Frameworks Evaluation

Page 40: Google test framework (public)

Thank you