aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/catch2/examples/111-Fix-PersistentFixture.cpp
blob: 2bef90ff7c10ba224f3872949be13bf4f884a9c7 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
//              Copyright Catch2 Authors
// Distributed under the Boost Software License, Version 1.0.
//   (See accompanying file LICENSE.txt or copy at
//        https://www.boost.org/LICENSE_1_0.txt)

// SPDX-License-Identifier: BSL-1.0

// Fixture.cpp

// Catch2 has three ways to express fixtures:
// - Sections
// - Traditional class-based fixtures that are created and destroyed on every
// partial run
// - Traditional class-based fixtures that are created at the start of a test
// case and destroyed at the end of a test case (this file)

// main() provided by linkage to Catch2WithMain

#include <catch2/catch_test_macros.hpp>

#include <thread>

class ClassWithExpensiveSetup {
public:
    ClassWithExpensiveSetup() {
        // Imagine some really expensive set up here.
        // e.g.
        // setting up a D3D12/Vulkan Device,
        // connecting to a database,
        // loading a file
        // etc etc etc
        std::this_thread::sleep_for( std::chrono::seconds( 2 ) );
    }

    ~ClassWithExpensiveSetup() noexcept {
        // We can do any clean up of the expensive class in the destructor
        // e.g.
        // destroy D3D12/Vulkan Device,
        // disconnecting from a database,
        // release file handle
        // etc etc etc
        std::this_thread::sleep_for( std::chrono::seconds( 1 ) );
    }

    int getInt() const { return 42; }
};

struct MyFixture {

    // The test case member function is const.
    // Therefore we need to mark any member of the fixture
    // that needs to mutate as mutable.
    mutable int myInt = 0;
    ClassWithExpensiveSetup expensive;
};

// Only one object of type MyFixture will be instantiated for the run
// of this test case even though there are two leaf sections.
// This is useful if your test case requires an object that is
// expensive to create and could be reused for each partial run of the
// test case.
TEST_CASE_PERSISTENT_FIXTURE( MyFixture, "Tests with MyFixture" ) {

    const int val = myInt++;

    SECTION( "First partial run" ) {
        const auto otherValue = expensive.getInt();
        REQUIRE( val == 0 );
        REQUIRE( otherValue == 42 );
    }

    SECTION( "Second partial run" ) { REQUIRE( val == 1 ); }
}