/*
* Scope Guard
* Copyright (C) 2017 offa
*
* This file is part of Scope Guard.
*
* Scope Guard is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Scope Guard is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Scope Guard. If not, see .
*/
#include "unique_resource.h"
#include
#include
using Handle = int;
using PtrHandle = std::add_pointer_t;
TEST_CASE("deleter called on destruction", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
static_cast(guard);
}
REQUIRE(calls == 1);
}
TEST_CASE("deleter is not called if released", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
guard.release();
}
REQUIRE(calls == 0);
}
TEST_CASE("deleter not called if checked valid", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource_checked(Handle{3}, Handle{6}, [&calls](auto) { ++calls; });
static_cast(guard);
}
REQUIRE(calls == 1);
}
TEST_CASE("deleter not called if checked invalid", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource_checked(Handle{3}, Handle{3}, [&calls](auto) { ++calls; });
static_cast(guard);
}
REQUIRE(calls == 0);
}
TEST_CASE("release returns reference to resource", "[UniqueResource]")
{
auto guard = sr::unique_resource(Handle{3}, [](auto) { });
const auto result = guard.release();
REQUIRE(3 == result);
}
TEST_CASE("move releases moved-from object", "[UniqueResource]")
{
std::size_t calls{0};
{
auto movedFrom = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
auto guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 1);
}
TEST_CASE("move transfers state", "[UniqueResource]")
{
std::size_t calls{0};
{
auto movedFrom = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
auto guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 1);
}
TEST_CASE("move transfers state if released", "[UniqueResource]")
{
std::size_t calls{0};
{
auto movedFrom = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
movedFrom.release();
auto guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 0);
}
TEST_CASE("move assignment releases moved-from object", "[UniqueResource]")
{
std::size_t calls{0};
std::function del = [&calls](auto) { ++calls; };
{
auto movedFrom = sr::unique_resource(Handle{3}, del);
auto guard = sr::unique_resource(Handle{3}, del);
guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 2);
}
TEST_CASE("move assignment transfers state", "[UniqueResource]")
{
std::size_t calls{0};
std::function del = [&calls](auto) { ++calls; };
{
auto movedFrom = sr::unique_resource(Handle{3}, del);
auto guard = sr::unique_resource(Handle{3}, del);
guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 2);
}
TEST_CASE("move assignment transfers state if released", "[UniqueResource]")
{
std::size_t calls{0};
std::function del = [&calls](auto) { ++calls; };
{
auto movedFrom = sr::unique_resource(Handle{3}, del);
movedFrom.release();
auto guard = sr::unique_resource(Handle{3}, del);
guard = std::move(movedFrom);
static_cast(guard);
}
REQUIRE(calls == 1);
}
TEST_CASE("no exception propagation from deleter", "[UniqueResource]")
{
REQUIRE_NOTHROW([] {
auto guard = sr::unique_resource(Handle{3}, [](auto) { throw "Don't propagate this!"; });
static_cast(guard);
}());
}
TEST_CASE("invoke executes deleter on resource", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto h)
{
REQUIRE(h == 3);
++calls;
});
guard.invoke();
}
REQUIRE(calls == 1);
}
TEST_CASE("invoke executes only multiple times if again strategy", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto h) { if( h == 3 ) { ++calls;} });
guard.invoke(sr::invoke_it::again);
guard.invoke(sr::invoke_it::again);
}
REQUIRE(calls == 3);
}
TEST_CASE("invoke does nothing if released", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
guard.release();
guard.invoke(sr::invoke_it::once);
}
REQUIRE(calls == 0);
}
TEST_CASE("invoke executes after release if again strategy", "[UniqueResource]")
{
std::size_t calls{0};
{
auto guard = sr::unique_resource(Handle{3}, [&calls](auto) { ++calls; });
guard.release();
guard.invoke(sr::invoke_it::again);
}
REQUIRE(calls == 1);
}
TEST_CASE("invoke does not propagate exception", "[UniqueResource]")
{
auto guard = sr::unique_resource(Handle{3}, [](auto) { throw "Don't propagate this!"; });
REQUIRE_NOTHROW(guard.invoke());
}
TEST_CASE("reset releases old ressource", "[UniqueResource]")
{
std::size_t calls{0};
{
auto d = [&calls](auto v)
{
if( calls == 0 )
{
REQUIRE(v == 3);
}
else
{
REQUIRE(v == 7);
}
++calls;
};
auto guard = sr::unique_resource(Handle{3}, d);
guard.reset(Handle{7});
}
REQUIRE(calls == 2);
}
TEST_CASE("reset sets ressource", "[UniqueResource]")
{
auto guard = sr::unique_resource(Handle{3}, [](auto) { });
guard.reset(Handle{7});
REQUIRE(guard.get() == 7);
}
TEST_CASE("get accesses ressource", "[UniqueResource]")
{
auto guard = sr::unique_resource(Handle{3}, [](auto) { });
REQUIRE(guard.get() == 3);
}
TEST_CASE("conversion operator", "[UniqueResource]")
{
auto guard = sr::unique_resource(Handle{3}, [](auto) { });
const auto& ref = guard;
REQUIRE(ref == 3);
}
TEST_CASE("pointer access operator" "[UniqueResource]")
{
const auto p = std::make_pair(3, 4);
auto guard = sr::unique_resource(&p, [](auto) { });
const auto x = guard.operator->();
REQUIRE(x->first == 3);
}
TEST_CASE("dereference operator", "[UniqueResource]")
{
Handle h{4};
auto guard = sr::unique_resource(PtrHandle{&h}, [](auto) { });
const auto x = guard.operator*();
REQUIRE(x == 4);
}
TEST_CASE("deleter access", "[UniqueResource]")
{
std::size_t value{0};
auto guard = sr::unique_resource(Handle{3}, [&value](auto v) { value = v; });
REQUIRE(value == 0);
guard.get_deleter()(6);
REQUIRE(value == 6);
}