/*
* 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 "scope_success.h"
#include "CallMocks.h"
#include
using namespace mock;
using namespace trompeloeil;
namespace
{
CallMock m;
void deleter()
{
m.deleter();
}
}
TEST_CASE("exit function called on destruction", "[ScopeSuccess]")
{
REQUIRE_CALL(m, deleter());
auto guard = sr::make_scope_success(deleter);
static_cast(guard);
}
TEST_CASE("exit function lambda called on destruction", "[ScopeSuccess]")
{
CallMock cm;
REQUIRE_CALL(cm, deleter());
auto guard = sr::make_scope_success([&cm] { cm.deleter(); });
static_cast(guard);
}
TEST_CASE("exit function not called and rethrow on copy exception", "[ScopeSuccess]")
{
REQUIRE_THROWS([] {
const ThrowOnCopyMock noMove;
REQUIRE_CALL(noMove, deleter());
sr::scope_success guard{noMove};
}());
}
TEST_CASE("exit function is not called if released", "[ScopeSuccess]")
{
REQUIRE_CALL(m, deleter()).TIMES(0);
auto guard = sr::make_scope_success(deleter);
guard.release();
}
TEST_CASE("move releases moved-from object", "[ScopeSuccess]")
{
REQUIRE_CALL(m, deleter());
auto movedFrom = sr::make_scope_success(deleter);
auto guard = std::move(movedFrom);
static_cast(guard);
}
TEST_CASE("move with copy init releases moved-from object", "[ScopeSuccess]")
{
CallMock mock;
const NotNothrowMoveMock notNothrow{&mock};
REQUIRE_CALL(mock, deleter());
sr::scope_success movedFrom{notNothrow};
auto guard = std::move(movedFrom);
}
TEST_CASE("move transfers state", "[ScopeSuccess]")
{
REQUIRE_CALL(m, deleter());
auto movedFrom = sr::make_scope_success(deleter);
auto guard = std::move(movedFrom);
static_cast(guard);
}
TEST_CASE("move transfers state if released", "[ScopeSuccess]")
{
REQUIRE_CALL(m, deleter()).TIMES(0);
auto movedFrom = sr::make_scope_success(deleter);
movedFrom.release();
auto guard = std::move(movedFrom);
static_cast(guard);
}
TEST_CASE("exit function not called on exception", "[ScopeFail]")
{
try
{
auto guard = sr::make_scope_success(deleter);
static_cast(guard);
throw 3;
}
catch( ... )
{
}
}
TEST_CASE("exit function called on pending exception", "[ScopeFail]")
{
try
{
throw 3;
}
catch( ... )
{
REQUIRE_CALL(m, deleter());
auto guard = sr::make_scope_success(deleter);
static_cast(guard);
}
}