aboutsummaryrefslogtreecommitdiffstats
path: root/config/src/tests/subscription/subscription.cpp
blob: 906457ae668cbe8301eeda4767ef5f78a47d6bdb (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
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
// Copyright Vespa.ai. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
#include <vespa/vespalib/testkit/test_kit.h>
#include <vespa/config/common/misc.h>
#include <vespa/config/common/configholder.h>
#include <vespa/config/subscription/configsubscription.h>
#include <config-my.h>
#include <thread>

using namespace config;

namespace {

struct SourceFixture
{
    int numClose;
    int numGetConfig;
    int numReload;
    SourceFixture()
        : numClose(0),
          numGetConfig(0),
          numReload(0)
    { }
};

struct MySource : public Source
{
    MySource(SourceFixture * src)
        : source(src)
    {}

    void getConfig() override { source->numGetConfig++; }
    void close() override { source->numClose++; }
    void reload(int64_t gen) override { (void) gen; source->numReload++; }

    SourceFixture * source;
};

struct SubscriptionFixture
{
    std::shared_ptr<IConfigHolder> holder;
    ConfigSubscription sub;
    SourceFixture src;
    SubscriptionFixture(const ConfigKey & key)
        : holder(new ConfigHolder()),
          sub(0, key, holder, std::make_unique<MySource>(&src))
    {
    }
};

vespalib::steady_time
deadline(vespalib::duration timeout) {
    return vespalib::steady_clock::now() + timeout;
}
}

TEST_FF("requireThatKeyIsReturned", ConfigKey("foo", "bar", "bim", "boo"), SubscriptionFixture(f1))
{
    ASSERT_TRUE(f1 == f2.sub.getKey());
}

TEST_F("requireThatUpdateReturns", SubscriptionFixture(ConfigKey::create<MyConfig>("myid")))
{
    f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), 1, 1));
    ASSERT_TRUE(f1.sub.nextUpdate(0, deadline(0ms)));
    ASSERT_TRUE(f1.sub.hasChanged());
    ASSERT_EQUAL(1, f1.sub.getGeneration());
}

TEST_F("requireThatNextUpdateBlocks", SubscriptionFixture(ConfigKey::create<MyConfig>("myid")))
{
    ASSERT_FALSE(f1.sub.nextUpdate(0, deadline(0ms)));
    f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), 1, 1));
    vespalib::Timer timer;
    ASSERT_FALSE(f1.sub.nextUpdate(1, deadline(500ms)));
    ASSERT_TRUE(timer.elapsed() > 400ms);
}

TEST_MT_F("requireThatNextUpdateReturnsWhenNotified", 2, SubscriptionFixture(ConfigKey::create<MyConfig>("myid")))
{
    if (thread_id == 0) {
        vespalib::Timer timer;
        f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), 1, 1));
        ASSERT_TRUE(f1.sub.nextUpdate(2, deadline(5000ms)));
        ASSERT_TRUE(timer.elapsed() > 200ms);
    } else {
        std::this_thread::sleep_for(500ms);
        f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), 1, 1));
    }
}


TEST_MT_F("requireThatNextUpdateReturnsInterrupted", 2, SubscriptionFixture(ConfigKey::create<MyConfig>("myid")))
{
    if (thread_id == 0) {
        vespalib::Timer timer;
        f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), 1, 1));
        ASSERT_TRUE(f1.sub.nextUpdate(1, deadline(5000ms)));
        ASSERT_TRUE(timer.elapsed() > 300ms);
    } else {
        std::this_thread::sleep_for(500ms);
        f1.sub.close();
    }
}

TEST_F("Require that isChanged takes generation into account", SubscriptionFixture(ConfigKey::create<MyConfig>("myid")))
{
    f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(StringVector(), "a"), true, 1));
    ASSERT_TRUE(f1.sub.nextUpdate(0, deadline(0ms)));
    f1.sub.flip();
    ASSERT_EQUAL(1, f1.sub.getLastGenerationChanged());
    f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(StringVector(), "b"), true, 2));
    ASSERT_TRUE(f1.sub.nextUpdate(1, deadline(0ms)));
    f1.sub.flip();
    ASSERT_EQUAL(2, f1.sub.getLastGenerationChanged());
    f1.holder->handle(std::make_unique<ConfigUpdate>(ConfigValue(), false, 3));
    ASSERT_TRUE(f1.sub.nextUpdate(2, deadline(0ms)));
    f1.sub.flip();
    ASSERT_EQUAL(2, f1.sub.getLastGenerationChanged());
}

TEST_MAIN() { TEST_RUN_ALL(); }