1
0
mirror of https://github.com/vcmi/vcmi.git synced 2025-01-24 03:47:18 +02:00
vcmi/test/spells/TargetConditionTest.cpp

309 lines
11 KiB
C++
Raw Normal View History

Spells configuration version 2 (effect-based) * Indirect spell effects loading * Json serializer improvements * spell->canBeCastAt do not allow useless cast for any spell * Added proxy caster class for spell-created obstacles * Handle damage from spell-created obstacles inside mechanics * Experimental GameState integration/regression tests * Ignore mod settings and load only "vcmi" mod when running tests * fixed https://bugs.vcmi.eu/view.php?id=2765 (with tests) * Huge improvements of BattleAI regarding spell casts * AI can cast almost any combat spell except TELEPORT, SACRIFICE and obstacle placement spells. * Possible fix for https://bugs.vcmi.eu/view.php?id=1811 * CStack factored out to several classes * [Battle] Allowed RETURN_AFTER_STRIKE effect on server side to be optional * [Battle] Allowed BattleAction have multiple destinations * [Spells] Converted limit|immunity to target condition * [Spells] Use partial configuration reload for backward compatibility handling * [Tests] Started tests for CUnitState * Partial fixes of fire shield effect * [Battle] Do HP calculations in 64 bits * [BattleAI] Use threading for spell cast evaluation * [BattleAI] Made AI be able to evaluate modified turn order (on hypothetical battle state) * Implemented https://bugs.vcmi.eu/view.php?id=2811 * plug rare freeze when hypnotized unit shots vertically * Correctly apply ONLY_MELEE_FIGHT / ONLY_DISTANCE_FIGHT for unit damage, attack & defense * [BattleAI] Try to not waste a cast if battle is actually won already * Extended JsonSerializeFormat API * fixed https://bugs.vcmi.eu/view.php?id=2847 * Any unit effect can be now chained (not only damage like Chain Lightning) ** only damage effect for now actually uses "chainFactor" * Possible quick fix for https://bugs.vcmi.eu/view.php?id=2860
2017-07-20 07:08:49 +03:00
/*
* TargetConditionTest.cpp, part of VCMI engine
*
* Authors: listed in file AUTHORS in main folder
*
* License: GNU General Public License v2.0 or later
* Full text of license available in license.txt file, in main folder
*
*/
#include "StdInc.h"
#include <vstd/RNG.h>
#include "../../lib/NetPacksBase.h"
#include "../../lib/spells/TargetCondition.h"
#include "../../lib/serializer/JsonDeserializer.h"
Spells configuration version 2 (effect-based) * Indirect spell effects loading * Json serializer improvements * spell->canBeCastAt do not allow useless cast for any spell * Added proxy caster class for spell-created obstacles * Handle damage from spell-created obstacles inside mechanics * Experimental GameState integration/regression tests * Ignore mod settings and load only "vcmi" mod when running tests * fixed https://bugs.vcmi.eu/view.php?id=2765 (with tests) * Huge improvements of BattleAI regarding spell casts * AI can cast almost any combat spell except TELEPORT, SACRIFICE and obstacle placement spells. * Possible fix for https://bugs.vcmi.eu/view.php?id=1811 * CStack factored out to several classes * [Battle] Allowed RETURN_AFTER_STRIKE effect on server side to be optional * [Battle] Allowed BattleAction have multiple destinations * [Spells] Converted limit|immunity to target condition * [Spells] Use partial configuration reload for backward compatibility handling * [Tests] Started tests for CUnitState * Partial fixes of fire shield effect * [Battle] Do HP calculations in 64 bits * [BattleAI] Use threading for spell cast evaluation * [BattleAI] Made AI be able to evaluate modified turn order (on hypothetical battle state) * Implemented https://bugs.vcmi.eu/view.php?id=2811 * plug rare freeze when hypnotized unit shots vertically * Correctly apply ONLY_MELEE_FIGHT / ONLY_DISTANCE_FIGHT for unit damage, attack & defense * [BattleAI] Try to not waste a cast if battle is actually won already * Extended JsonSerializeFormat API * fixed https://bugs.vcmi.eu/view.php?id=2847 * Any unit effect can be now chained (not only damage like Chain Lightning) ** only damage effect for now actually uses "chainFactor" * Possible quick fix for https://bugs.vcmi.eu/view.php?id=2860
2017-07-20 07:08:49 +03:00
#include "mock/mock_spells_Mechanics.h"
#include "mock/mock_BonusBearer.h"
#include "mock/mock_battle_Unit.h"
namespace test
{
using namespace ::spells;
using namespace ::testing;
class ItemMock : public TargetConditionItem
{
public:
MOCK_CONST_METHOD2(isReceptive, bool(const Mechanics *, const battle::Unit *));
MOCK_METHOD1(setInverted, void(bool));
MOCK_METHOD1(setExclusive, void(bool));
MOCK_CONST_METHOD0(isExclusive, bool());
};
class FactoryMock : public TargetConditionItemFactory
{
public:
MOCK_CONST_METHOD0(createAbsoluteLevel, Object());
MOCK_CONST_METHOD0(createAbsoluteSpell, Object());
MOCK_CONST_METHOD0(createElemental, Object());
MOCK_CONST_METHOD0(createNormalLevel, Object());
MOCK_CONST_METHOD0(createNormalSpell, Object());
2023-03-23 12:01:13 +03:00
MOCK_CONST_METHOD1(createFromJsonStruct, Object(const JsonNode &));
Spells configuration version 2 (effect-based) * Indirect spell effects loading * Json serializer improvements * spell->canBeCastAt do not allow useless cast for any spell * Added proxy caster class for spell-created obstacles * Handle damage from spell-created obstacles inside mechanics * Experimental GameState integration/regression tests * Ignore mod settings and load only "vcmi" mod when running tests * fixed https://bugs.vcmi.eu/view.php?id=2765 (with tests) * Huge improvements of BattleAI regarding spell casts * AI can cast almost any combat spell except TELEPORT, SACRIFICE and obstacle placement spells. * Possible fix for https://bugs.vcmi.eu/view.php?id=1811 * CStack factored out to several classes * [Battle] Allowed RETURN_AFTER_STRIKE effect on server side to be optional * [Battle] Allowed BattleAction have multiple destinations * [Spells] Converted limit|immunity to target condition * [Spells] Use partial configuration reload for backward compatibility handling * [Tests] Started tests for CUnitState * Partial fixes of fire shield effect * [Battle] Do HP calculations in 64 bits * [BattleAI] Use threading for spell cast evaluation * [BattleAI] Made AI be able to evaluate modified turn order (on hypothetical battle state) * Implemented https://bugs.vcmi.eu/view.php?id=2811 * plug rare freeze when hypnotized unit shots vertically * Correctly apply ONLY_MELEE_FIGHT / ONLY_DISTANCE_FIGHT for unit damage, attack & defense * [BattleAI] Try to not waste a cast if battle is actually won already * Extended JsonSerializeFormat API * fixed https://bugs.vcmi.eu/view.php?id=2847 * Any unit effect can be now chained (not only damage like Chain Lightning) ** only damage effect for now actually uses "chainFactor" * Possible quick fix for https://bugs.vcmi.eu/view.php?id=2860
2017-07-20 07:08:49 +03:00
MOCK_CONST_METHOD3(createConfigurable, Object(std::string, std::string, std::string));
MOCK_CONST_METHOD0(createReceptiveFeature, Object());
MOCK_CONST_METHOD0(createImmunityNegation, Object());
};
class TargetConditionTest : public Test
{
public:
TargetCondition subject;
StrictMock<MechanicsMock> mechanicsMock;
StrictMock<UnitMock> targetMock;
NiceMock<FactoryMock> factoryMock;
void setupSubject(const JsonNode & configuration)
{
JsonDeserializer handler(nullptr, configuration);
subject.serializeJson(handler, &factoryMock);
}
void redirectFactoryToStub()
{
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*itemStub, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillRepeatedly(Return(true));
EXPECT_CALL(*itemStub, isExclusive()).WillRepeatedly(Return(false));
ON_CALL(factoryMock, createAbsoluteLevel()).WillByDefault(Return(itemStub));
ON_CALL(factoryMock, createAbsoluteSpell()).WillByDefault(Return(itemStub));
ON_CALL(factoryMock, createElemental()).WillByDefault(Return(itemStub));
ON_CALL(factoryMock, createNormalLevel()).WillByDefault(Return(itemStub));
ON_CALL(factoryMock, createNormalSpell()).WillByDefault(Return(itemStub));
auto negationItemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*negationItemStub, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillRepeatedly(Return(false));
ON_CALL(factoryMock, createReceptiveFeature()).WillByDefault(Return(negationItemStub));
ON_CALL(factoryMock, createImmunityNegation()).WillByDefault(Return(negationItemStub));
}
TargetConditionTest(){}
~TargetConditionTest(){}
protected:
void SetUp() override
{
}
private:
};
TEST_F(TargetConditionTest, SerializesCorrectly)
{
redirectFactoryToStub();
auto normalInvertedItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*normalInvertedItem, isReceptive(_,_)).WillRepeatedly(Return(true));
EXPECT_CALL(*normalInvertedItem, setInverted(Eq(true))).Times(1);
EXPECT_CALL(*normalInvertedItem, setExclusive(Eq(true))).Times(1);
EXPECT_CALL(*normalInvertedItem, isExclusive()).WillRepeatedly(Return(true));
EXPECT_CALL(factoryMock, createConfigurable(Eq(""), Eq("bonus"), Eq("NON_LIVING"))).WillOnce(Return(normalInvertedItem));
auto absoluteItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*absoluteItem, isReceptive(_,_)).WillRepeatedly(Return(true));
EXPECT_CALL(*absoluteItem, setInverted(Eq(false))).Times(1);
EXPECT_CALL(*absoluteItem, setExclusive(Eq(false))).Times(1);
EXPECT_CALL(*absoluteItem, isExclusive()).WillRepeatedly(Return(true));
EXPECT_CALL(factoryMock, createConfigurable(Eq(""), Eq("bonus"), Eq("SIEGE_WEAPON"))).WillOnce(Return(absoluteItem));
auto normalItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*normalItem, isReceptive(_,_)).WillRepeatedly(Return(true));
EXPECT_CALL(*normalItem, setInverted(Eq(false))).Times(1);
EXPECT_CALL(*normalItem, setExclusive(Eq(true))).Times(1);
EXPECT_CALL(*normalItem, isExclusive()).WillRepeatedly(Return(true));
EXPECT_CALL(factoryMock, createConfigurable(Eq(""), Eq("bonus"), Eq("UNDEAD"))).WillOnce(Return(normalItem));
JsonNode config(JsonNode::JsonType::DATA_STRUCT);
config["noneOf"]["bonus.NON_LIVING"].String() = "normal";
config["anyOf"]["bonus.SIEGE_WEAPON"].String() = "absolute";
config["allOf"]["bonus.UNDEAD"].String() = "normal";
setupSubject(config);
EXPECT_THAT(subject.normal, Contains(normalInvertedItem));
EXPECT_THAT(subject.normal, Contains(normalItem));
EXPECT_THAT(subject.absolute, Contains(absoluteItem));
}
TEST_F(TargetConditionTest, CreatesSpecialConditions)
{
redirectFactoryToStub();
EXPECT_CALL(factoryMock, createAbsoluteLevel()).Times(1);
EXPECT_CALL(factoryMock, createAbsoluteSpell()).Times(1);
EXPECT_CALL(factoryMock, createElemental()).Times(1);
EXPECT_CALL(factoryMock, createNormalLevel()).Times(1);
EXPECT_CALL(factoryMock, createNormalSpell()).Times(1);
EXPECT_CALL(factoryMock, createReceptiveFeature()).Times(1);
EXPECT_CALL(factoryMock, createImmunityNegation()).Times(1);
setupSubject(JsonNode());
}
TEST_F(TargetConditionTest, StoresAbsoluteLevelCondition)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createAbsoluteLevel()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.absolute, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresAbsoluteSpellCondition)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createAbsoluteSpell()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.absolute, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresElementalCondition)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createElemental()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.normal, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresNormalSpellCondition)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createNormalSpell()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.normal, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresNormalLevelCondition)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createNormalLevel()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.normal, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresReceptiveFeature)
{
2021-07-05 20:04:17 +03:00
redirectFactoryToStub();
Spells configuration version 2 (effect-based) * Indirect spell effects loading * Json serializer improvements * spell->canBeCastAt do not allow useless cast for any spell * Added proxy caster class for spell-created obstacles * Handle damage from spell-created obstacles inside mechanics * Experimental GameState integration/regression tests * Ignore mod settings and load only "vcmi" mod when running tests * fixed https://bugs.vcmi.eu/view.php?id=2765 (with tests) * Huge improvements of BattleAI regarding spell casts * AI can cast almost any combat spell except TELEPORT, SACRIFICE and obstacle placement spells. * Possible fix for https://bugs.vcmi.eu/view.php?id=1811 * CStack factored out to several classes * [Battle] Allowed RETURN_AFTER_STRIKE effect on server side to be optional * [Battle] Allowed BattleAction have multiple destinations * [Spells] Converted limit|immunity to target condition * [Spells] Use partial configuration reload for backward compatibility handling * [Tests] Started tests for CUnitState * Partial fixes of fire shield effect * [Battle] Do HP calculations in 64 bits * [BattleAI] Use threading for spell cast evaluation * [BattleAI] Made AI be able to evaluate modified turn order (on hypothetical battle state) * Implemented https://bugs.vcmi.eu/view.php?id=2811 * plug rare freeze when hypnotized unit shots vertically * Correctly apply ONLY_MELEE_FIGHT / ONLY_DISTANCE_FIGHT for unit damage, attack & defense * [BattleAI] Try to not waste a cast if battle is actually won already * Extended JsonSerializeFormat API * fixed https://bugs.vcmi.eu/view.php?id=2847 * Any unit effect can be now chained (not only damage like Chain Lightning) ** only damage effect for now actually uses "chainFactor" * Possible quick fix for https://bugs.vcmi.eu/view.php?id=2860
2017-07-20 07:08:49 +03:00
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createReceptiveFeature()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.negation, Contains(itemStub));
}
TEST_F(TargetConditionTest, StoresImmunityNegation)
{
redirectFactoryToStub();
auto itemStub = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(factoryMock, createImmunityNegation()).WillOnce(Return(itemStub));
setupSubject(JsonNode());
EXPECT_THAT(subject.negation, Contains(itemStub));
}
TEST_F(TargetConditionTest, ChecksAbsoluteCondition)
{
auto item = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*item, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*item, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.absolute.push_back(item);
EXPECT_TRUE(subject.isReceptive(&mechanicsMock, &targetMock));
}
TEST_F(TargetConditionTest, ChecksNegationCondition)
{
auto item = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*item, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.negation.push_back(item);
EXPECT_TRUE(subject.isReceptive(&mechanicsMock, &targetMock));
}
TEST_F(TargetConditionTest, ChecksNormalCondition)
{
auto item = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*item, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*item, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.normal.push_back(item);
EXPECT_TRUE(subject.isReceptive(&mechanicsMock, &targetMock));
}
TEST_F(TargetConditionTest, ChecksAbsoluteConditionFirst)
{
redirectFactoryToStub();
setupSubject(JsonNode());
auto absoluteItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*absoluteItem, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*absoluteItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(false));
subject.absolute.push_back(absoluteItem);
auto negationItem = std::make_shared<StrictMock<ItemMock>>();
subject.negation.push_back(negationItem);
auto normalItem = std::make_shared<StrictMock<ItemMock>>();
subject.normal.push_back(normalItem);
EXPECT_FALSE(subject.isReceptive(&mechanicsMock, &targetMock));
}
TEST_F(TargetConditionTest, ChecksNegationConditionSecond)
{
redirectFactoryToStub();
setupSubject(JsonNode());
auto absoluteItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*absoluteItem, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*absoluteItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.absolute.push_back(absoluteItem);
auto negationItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*negationItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.negation.push_back(negationItem);
auto normalItem = std::make_shared<StrictMock<ItemMock>>();
subject.normal.push_back(normalItem);
EXPECT_TRUE(subject.isReceptive(&mechanicsMock, &targetMock));
}
TEST_F(TargetConditionTest, ChecksNormalConditionThird)
{
redirectFactoryToStub();
setupSubject(JsonNode());
auto absoluteItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*absoluteItem, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*absoluteItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(true));
subject.absolute.push_back(absoluteItem);
auto negationItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*negationItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(false));
subject.negation.push_back(negationItem);
auto normalItem = std::make_shared<StrictMock<ItemMock>>();
EXPECT_CALL(*normalItem, isExclusive()).WillOnce(Return(true));
EXPECT_CALL(*normalItem, isReceptive(Eq(&mechanicsMock), Eq(&targetMock))).WillOnce(Return(false));
subject.normal.push_back(normalItem);
EXPECT_FALSE(subject.isReceptive(&mechanicsMock, &targetMock));
}
}