/** * Copyright 2015 Denis Blank * * Licensed under the Apache License, Version 2.0 (the "License"); * 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. */ #define CATCH_CONFIG_RUNNER #include "catch.hpp" #include "Continuable.h" void testNextGen(); void test_mockup(); void test_incubator(); int main(int argc, char* const argv[]) { testNextGen(); return 0; test_mockup(); test_incubator(); int const result = Catch::Session().run(argc, argv); // Attach breakpoint here ,-) return result; } template using cross_forward_t = typename std::conditional< std::is_rvalue_reference::value, typename std::decay::type&&, typename std::decay::type& >::type; template cross_forward_t cross_forward(V&& var) { return static_cast&&>(var); } struct TestContainer { std::shared_ptr ptr; }; template TestContainer extract(T&& c) { return TestContainer{cross_forward(c.ptr)}; } TEST_CASE("CrossForward tests", "[CrossForward]") { TestContainer con; con.ptr = std::make_shared(0); static_assert(std::is_same< cross_forward_t< TestContainer&&, std::unique_ptr& >, std::unique_ptr&&>::value, "cross_forward returns wrong type!"); static_assert(std::is_same< cross_forward_t< TestContainer&, std::unique_ptr& >, std::unique_ptr&>::value, "cross_forward returns wrong type!"); SECTION("forward l-value references") { extract(con); REQUIRE(con.ptr.get()); } SECTION("forward r-value references") { extract(std::move(con)); REQUIRE_FALSE(con.ptr.get()); } } Continuable http_request(std::string const& /*url*/) { return make_continuable([=](std::function&& callback) { // Do request... std::string result = "some HTTP content"; callback(std::move(result)); }); } struct ResultSet { std::size_t rows; }; Continuable mysql_query(std::string const& /*query*/) { return make_continuable([=](std::function&& callback) { callback(ResultSet{5}); }); } Continuable<> defect_continuable() { return make_continuable([=](std::function&& /*callback*/) { // Callback is never called }); } TEST_CASE("Continuable invocation on destruct", "[Continuable]") { bool invoked = false; std::function&&)> invokeable = [&](Callback<>&& callback) { invoked = true; callback(); }; SECTION("Continuables are not invoked before destruct") { Continuable<> continuable = make_continuable(std::move(invokeable)); REQUIRE_FALSE(invoked); } SECTION("Continuables are invoked on destruct") { make_continuable(std::move(invokeable)); REQUIRE(invoked); } SECTION("Continuables are not invoked after transferred") { Continuable<> continuable = make_continuable(std::move(invokeable)); { Continuable<> cache = std::move(continuable); REQUIRE_FALSE(invoked); } REQUIRE(invoked); } } TEST_CASE("Continuable continuation chaining using Continuable::then", "[Continuable]") { std::size_t invoked = 0; SECTION("Continuables result is evaluateable") { make_continuable([](Callback&& callback) { callback(12345); }) .then([&](int i) { if (i == 12345) ++invoked; }); REQUIRE(invoked == 1); } SECTION("Continuables are invalidated on chaining (no duplicated call) and order is ok") { http_request("http://github.com") .then([&](/*std::string url (use signature erasure)*/) { REQUIRE(invoked == 0); invoked = 1; }) .then([&] { REQUIRE(invoked == 1); invoked = 2; }) .then([&] { REQUIRE(invoked == 2); invoked = 3; return make_continuable([&](Callback&& callback) { REQUIRE(invoked == 3); invoked = 4; callback(5); }); }) .then([&](std::size_t t) { REQUIRE(invoked == 4); invoked = t; }); REQUIRE(invoked == 5); } SECTION("Continuation chains need a callback to continue") { defect_continuable() .then([&] { invoked++; }); REQUIRE(invoked == 0); } }