397 lines
11 KiB
C++
397 lines
11 KiB
C++
/*
|
|
* This file is open source software, licensed to you under the terms
|
|
* of the Apache License, Version 2.0 (the "License"). See the NOTICE file
|
|
* distributed with this work for additional information regarding copyright
|
|
* ownership. You may not use this file except in compliance with the License.
|
|
*
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
/*
|
|
* Copyright 2021 ScyllaDB
|
|
*/
|
|
|
|
#include <exception>
|
|
|
|
#include <boost/range/irange.hpp>
|
|
|
|
#include <seastar/testing/test_case.hh>
|
|
#include <seastar/testing/thread_test_case.hh>
|
|
|
|
#include <seastar/core/gate.hh>
|
|
#include <seastar/util/closeable.hh>
|
|
#include <seastar/core/loop.hh>
|
|
|
|
using namespace seastar;
|
|
|
|
class expected_exception : public std::runtime_error {
|
|
public:
|
|
expected_exception() : runtime_error("expected") {}
|
|
};
|
|
|
|
SEASTAR_TEST_CASE(deferred_close_test) {
|
|
return do_with(gate(), 0, 42, [] (gate& g, int& count, int& expected) {
|
|
return async([&] {
|
|
auto close_gate = deferred_close(g);
|
|
|
|
for (auto i = 0; i < expected; i++) {
|
|
(void)with_gate(g, [&count] {
|
|
++count;
|
|
});
|
|
}
|
|
}).then([&] {
|
|
// destroying close_gate should invoke g.close()
|
|
// and wait for all background continuations to complete
|
|
BOOST_REQUIRE(g.is_closed());
|
|
BOOST_REQUIRE_EQUAL(count, expected);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(move_deferred_close_test) {
|
|
return do_with(gate(), [] (gate& g) {
|
|
return async([&] {
|
|
auto close_gate = make_shared(deferred_close(g));
|
|
// g.close() should not be called when deferred_close is moved away
|
|
BOOST_REQUIRE(!g.is_closed());
|
|
}).then([&] {
|
|
// Before this test is exercised, gate::close() would run into a
|
|
// assert failure when leaving previous continuation, if gate::close()
|
|
// is called twice, so this test only verifies the behavior with the
|
|
// release build.
|
|
BOOST_REQUIRE(g.is_closed());
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(close_now_test) {
|
|
return do_with(gate(), 0, 42, [] (gate& g, int& count, int& expected) {
|
|
return async([&] {
|
|
auto close_gate = deferred_close(g);
|
|
|
|
for (auto i = 0; i < expected; i++) {
|
|
(void)with_gate(g, [&count] {
|
|
++count;
|
|
});
|
|
}
|
|
|
|
close_gate.close_now();
|
|
BOOST_REQUIRE(g.is_closed());
|
|
BOOST_REQUIRE_EQUAL(count, expected);
|
|
// gate must not be double-closed.
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(cancel_deferred_close_test) {
|
|
gate g;
|
|
{
|
|
auto close_gate = deferred_close(g);
|
|
close_gate.cancel();
|
|
}
|
|
g.check(); // should not throw
|
|
return make_ready_future<>();
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(with_closeable_test) {
|
|
return do_with(0, 42, [] (int& count, int& expected) {
|
|
return with_closeable(gate(), [&] (gate& g) {
|
|
for (auto i = 0; i < expected; i++) {
|
|
(void)with_gate(g, [&count] {
|
|
++count;
|
|
});
|
|
}
|
|
return 17;
|
|
}).then([&] (int res) {
|
|
// res should be returned by the function called
|
|
// by with_closeable.
|
|
BOOST_REQUIRE_EQUAL(res, 17);
|
|
// closing the gate should wait for
|
|
// all background continuations to complete
|
|
BOOST_REQUIRE_EQUAL(count, expected);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(with_closeable_exception_test) {
|
|
return do_with(0, 42, [] (int& count, int& expected) {
|
|
return with_closeable(gate(), [&] (gate& g) {
|
|
for (auto i = 0; i < expected; i++) {
|
|
(void)with_gate(g, [&count] {
|
|
++count;
|
|
});
|
|
}
|
|
throw expected_exception();
|
|
}).handle_exception_type([&] (const expected_exception&) {
|
|
// closing the gate should also happen when func throws,
|
|
// waiting for all background continuations to complete
|
|
BOOST_REQUIRE_EQUAL(count, expected);
|
|
});
|
|
});
|
|
}
|
|
|
|
namespace {
|
|
|
|
class count_stops {
|
|
int _count = -1;
|
|
int* _ptr = nullptr;
|
|
public:
|
|
count_stops(int* ptr = nullptr) noexcept
|
|
: _ptr(ptr ? ptr : &_count)
|
|
{
|
|
*_ptr = 0;
|
|
}
|
|
|
|
count_stops(count_stops&& o) noexcept {
|
|
std::exchange(_count, o._count);
|
|
if (o._ptr == &o._count) {
|
|
_ptr = &_count;
|
|
} else {
|
|
std::exchange(_ptr, o._ptr);
|
|
}
|
|
}
|
|
|
|
future<> stop() noexcept {
|
|
++*_ptr;
|
|
return make_ready_future<>();
|
|
}
|
|
|
|
int stopped() const noexcept {
|
|
return *_ptr;
|
|
}
|
|
};
|
|
|
|
} // anonymous namespace
|
|
|
|
SEASTAR_TEST_CASE(cancel_deferred_stop_test) {
|
|
count_stops cs;
|
|
{
|
|
auto stop = deferred_stop(cs);
|
|
stop.cancel();
|
|
}
|
|
BOOST_REQUIRE_EQUAL(cs.stopped(), 0);
|
|
return make_ready_future<>();
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(deferred_stop_test) {
|
|
return do_with(count_stops(), [] (count_stops& cs) {
|
|
return async([&] {
|
|
auto stop_counting = deferred_stop(cs);
|
|
}).then([&] {
|
|
// cs.stop() should be called when stop_counting is destroyed
|
|
BOOST_REQUIRE_EQUAL(cs.stopped(), 1);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(move_deferred_stop_test) {
|
|
return do_with(count_stops(), [] (count_stops& cs) {
|
|
return async([&] {
|
|
auto stop = make_shared(deferred_stop(cs));
|
|
}).then([&] {
|
|
// cs.stop() should be called once and only once
|
|
// when stop is destroyed
|
|
BOOST_REQUIRE_EQUAL(cs.stopped(), 1);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(stop_now_test) {
|
|
return do_with(count_stops(), [] (count_stops& cs) {
|
|
return async([&] {
|
|
auto stop_counting = deferred_stop(cs);
|
|
|
|
stop_counting.stop_now();
|
|
// cs.stop() should not be called again
|
|
// when stop_counting is destroyed
|
|
BOOST_REQUIRE_EQUAL(cs.stopped(), 1);
|
|
}).then([&] {
|
|
// cs.stop() should be called exactly once
|
|
BOOST_REQUIRE_EQUAL(cs.stopped(), 1);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(with_stoppable_test) {
|
|
return do_with(0, [] (int& stopped) {
|
|
return with_stoppable(count_stops(&stopped), [] (count_stops& cs) {
|
|
return 17;
|
|
}).then([&] (int res) {
|
|
// res should be returned by the function called
|
|
// by with_closeable.
|
|
BOOST_REQUIRE_EQUAL(res, 17);
|
|
// cs.stop() should be called before count_stops is destroyed
|
|
BOOST_REQUIRE_EQUAL(stopped, 1);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(with_stoppable_exception_test) {
|
|
return do_with(0, [] (int& stopped) {
|
|
return with_stoppable(count_stops(&stopped), [] (count_stops& cs) {
|
|
throw expected_exception();
|
|
}).handle_exception_type([&] (const expected_exception&) {
|
|
// cs.stop() should be called before count_stops is destroyed
|
|
// also when func throws
|
|
BOOST_REQUIRE_EQUAL(stopped, 1);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(move_open_gate_test) {
|
|
gate g1;
|
|
g1.enter();
|
|
// move an open gate
|
|
gate g2 = std::move(g1);
|
|
// the state in g1 should be moved into g2
|
|
BOOST_CHECK_EQUAL(g1.get_count(), 0);
|
|
BOOST_REQUIRE_EQUAL(g2.get_count(), 1);
|
|
g2.leave();
|
|
g2.close().get();
|
|
BOOST_CHECK(!g1.is_closed());
|
|
BOOST_CHECK(g2.is_closed());
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(move_closing_gate_test) {
|
|
gate g1;
|
|
g1.enter();
|
|
auto fut = g1.close();
|
|
// move a closing gate
|
|
gate g2 = std::move(g1);
|
|
BOOST_CHECK_EQUAL(g1.get_count(), 0);
|
|
BOOST_REQUIRE_EQUAL(g2.get_count(), 1);
|
|
g2.leave();
|
|
fut.get();
|
|
BOOST_CHECK(!g1.is_closed());
|
|
BOOST_CHECK(g2.is_closed());
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(move_closed_gate_test) {
|
|
gate g1;
|
|
g1.close().get();
|
|
// move a closed gate
|
|
gate g2 = std::move(g1);
|
|
BOOST_CHECK_EQUAL(g1.get_count(), 0);
|
|
BOOST_CHECK_EQUAL(g2.get_count(), 0);
|
|
BOOST_CHECK(!g1.is_closed());
|
|
BOOST_CHECK(g2.is_closed());
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_basic_test) {
|
|
gate g;
|
|
auto gh = g.hold();
|
|
auto fut = g.close();
|
|
BOOST_CHECK(!fut.available());
|
|
gh.release();
|
|
fut.get();
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_closed_test) {
|
|
gate g;
|
|
g.close().get();
|
|
BOOST_REQUIRE_THROW(g.hold(), gate_closed_exception);
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_move_test) {
|
|
gate g;
|
|
auto gh0 = g.hold();
|
|
auto fut = g.close();
|
|
BOOST_CHECK(!fut.available());
|
|
auto gh1 = std::move(gh0);
|
|
BOOST_CHECK(!fut.available());
|
|
gh1.release();
|
|
fut.get();
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_copy_test) {
|
|
gate g;
|
|
auto gh0 = g.hold();
|
|
auto gh1 = gh0;
|
|
auto fut = g.close();
|
|
BOOST_CHECK(!fut.available());
|
|
gh0.release();
|
|
BOOST_CHECK(!fut.available());
|
|
gh1.release();
|
|
fut.get();
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_copy_and_move_test) {
|
|
gate g0;
|
|
auto gh00 = g0.hold();
|
|
auto gh01 = gh00;
|
|
auto fut0 = g0.close();
|
|
BOOST_CHECK(!fut0.available());
|
|
gate g1;
|
|
auto gh1 = g1.hold();
|
|
auto fut1 = g1.close();
|
|
BOOST_CHECK(!fut1.available());
|
|
gh01.release();
|
|
BOOST_CHECK(!fut0.available());
|
|
BOOST_CHECK(!fut1.available());
|
|
gh00 = std::move(gh1);
|
|
fut0.get();
|
|
BOOST_CHECK(!fut1.available());
|
|
gh00.release();
|
|
fut1.get();
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_copy_after_close_test) {
|
|
gate g;
|
|
auto gh0 = g.hold();
|
|
auto fut = g.close();
|
|
BOOST_CHECK(g.is_closed());
|
|
gate::holder gh1 = gh0;
|
|
BOOST_CHECK(!fut.available());
|
|
gh0.release();
|
|
BOOST_CHECK(!fut.available());
|
|
gh1.release();
|
|
fut.get();
|
|
}
|
|
|
|
SEASTAR_TEST_CASE(gate_holder_parallel_copy_test) {
|
|
constexpr int expected = 42;
|
|
return do_with(0, [expected] (int& count) {
|
|
return with_closeable(gate(), [&] (gate& g) {
|
|
auto gh = g.hold();
|
|
// Copying the gate::holder in the lambda below should keep it open
|
|
// until all instances complete
|
|
(void)parallel_for_each(boost::irange(0, expected), [&count, gh = gh] (int) {
|
|
count++;
|
|
return make_ready_future<>();
|
|
});
|
|
return 17;
|
|
}).then([&, expected] (int res) {
|
|
// res should be returned by the function called
|
|
// by with_closeable.
|
|
BOOST_REQUIRE_EQUAL(res, 17);
|
|
// closing the gate should wait for
|
|
// all background continuations to complete
|
|
BOOST_REQUIRE_EQUAL(count, expected);
|
|
});
|
|
});
|
|
}
|
|
|
|
SEASTAR_THREAD_TEST_CASE(gate_holder_try_close_test) {
|
|
gate g;
|
|
auto gh0 = g.try_hold();
|
|
BOOST_CHECK(gh0.has_value());
|
|
auto fut = g.close();
|
|
BOOST_CHECK(g.is_closed());
|
|
auto failed_gh = g.try_hold();
|
|
BOOST_CHECK(!failed_gh.has_value());
|
|
auto gh1 = std::move(gh0);
|
|
BOOST_CHECK(!fut.available());
|
|
gh0.reset();
|
|
BOOST_CHECK(!fut.available());
|
|
gh1.reset();
|
|
fut.get();
|
|
}
|