From 9f882824a09a7563f343505809aac020906a0d47 Mon Sep 17 00:00:00 2001 From: Oleg Agafonov Date: Wed, 11 Aug 2021 12:43:13 +0400 Subject: [PATCH] * Gain ability effect: fixed rollback error with AI games and too many cards (related to d202278ccd6876f8113d26951e6bd018f6e9b792); --- .../ConcurrentModificationExceptionTest.java | 51 +++++++++++++ .../test/cards/continuous/WarpworldTest.java | 71 ------------------- .../continuous/GainAbilityAllEffect.java | 12 ---- .../GainAbilityControlledEffect.java | 14 ---- .../continuous/LoseAbilityAllEffect.java | 12 ---- 5 files changed, 51 insertions(+), 109 deletions(-) create mode 100644 Mage.Tests/src/test/java/org/mage/test/cards/continuous/ConcurrentModificationExceptionTest.java delete mode 100644 Mage.Tests/src/test/java/org/mage/test/cards/continuous/WarpworldTest.java diff --git a/Mage.Tests/src/test/java/org/mage/test/cards/continuous/ConcurrentModificationExceptionTest.java b/Mage.Tests/src/test/java/org/mage/test/cards/continuous/ConcurrentModificationExceptionTest.java new file mode 100644 index 0000000000..2754728439 --- /dev/null +++ b/Mage.Tests/src/test/java/org/mage/test/cards/continuous/ConcurrentModificationExceptionTest.java @@ -0,0 +1,51 @@ +package org.mage.test.cards.continuous; + +import mage.constants.PhaseStep; +import mage.constants.Zone; +import org.junit.Ignore; +import org.junit.Test; +import org.mage.test.serverside.base.CardTestPlayerBaseWithAIHelps; + +public class ConcurrentModificationExceptionTest extends CardTestPlayerBaseWithAIHelps { + + + // see details here: https://github.com/magefree/mage/commit/d202278ccd6876f8113d26951e6bd018f6e9b792 + + @Ignore // debug only: too slow, only for manual run (use multi run to ensure) + @Test + public void massWarpWorld() { + removeAllCardsFromLibrary(playerA); + + addCard(Zone.BATTLEFIELD, playerA, "Prosperous Innkeeper"); + addCard(Zone.BATTLEFIELD, playerA, "Tireless Tracker"); + addCard(Zone.BATTLEFIELD, playerA, "Gilded Goose"); + addCard(Zone.BATTLEFIELD, playerA, "Galazeth Prismari"); + addCard(Zone.BATTLEFIELD, playerA, "Wood Elves"); + addCard(Zone.BATTLEFIELD, playerA, "Nesting Dragon"); + addCard(Zone.BATTLEFIELD, playerA, "Primeval Titan"); + addCard(Zone.BATTLEFIELD, playerA, "Eternal Witness"); + addCard(Zone.BATTLEFIELD, playerA, "Westvale Abbey"); + addCard(Zone.BATTLEFIELD, playerA, "Gaea's Cradle"); + addCard(Zone.BATTLEFIELD, playerA, "Castle Garenbrig"); + addCard(Zone.BATTLEFIELD, playerA, "Nyxbloom Ancient"); + addCard(Zone.BATTLEFIELD, playerA, "Huntmaster of the Fells"); + addCard(Zone.BATTLEFIELD, playerA, "Xenagos, the Reveler"); + addCard(Zone.BATTLEFIELD, playerA, "Thromok the Insatiable"); + addCard(Zone.BATTLEFIELD, playerA, "The World Tree"); + addCard(Zone.BATTLEFIELD, playerA, "Nylea, Keen-Eyed"); + addCard(Zone.BATTLEFIELD, playerA, "Kogla, the Titan Ape"); + addCard(Zone.BATTLEFIELD, playerA, "Acidic Slime"); + addCard(Zone.BATTLEFIELD, playerA, "Druid Class"); + addCard(Zone.BATTLEFIELD, playerA, "Mountain", 10); + addCard(Zone.HAND, playerA, "Warp World"); + + // simple test + castSpell(1, PhaseStep.PRECOMBAT_MAIN, playerA, "Warp World"); + aiPlayStep(1, PhaseStep.PRECOMBAT_MAIN, playerA); + + //setStrictChooseMode(true); // need AI while cards adding + setStopAt(1, PhaseStep.BEGIN_COMBAT); + execute(); + assertAllCommandsUsed(); + } +} diff --git a/Mage.Tests/src/test/java/org/mage/test/cards/continuous/WarpworldTest.java b/Mage.Tests/src/test/java/org/mage/test/cards/continuous/WarpworldTest.java deleted file mode 100644 index bed01da58b..0000000000 --- a/Mage.Tests/src/test/java/org/mage/test/cards/continuous/WarpworldTest.java +++ /dev/null @@ -1,71 +0,0 @@ -package org.mage.test.cards.continuous; - -import mage.constants.PhaseStep; -import mage.constants.Zone; -import org.junit.Assert; -import org.junit.Test; -import org.mage.test.serverside.base.CardTestPlayerBase; - -import java.util.ConcurrentModificationException; - -// goal is to test the concurrentmodification exception related to continuous effects -public class WarpworldTest extends CardTestPlayerBase { - - private final String innkeeper = "Prosperous Innkeeper"; - private final String tracker = "Tireless Tracker"; - private final String goose = "Gilded Goose"; - private final String galazeth_prismari = "Galazeth Prismari"; - private final String wood_elves = "Wood Elves"; - private final String nesting_dragon = "Nesting Dragon"; - private final String primeval_titan = "Primeval Titan"; - private final String eternal_witness = "Eternal Witness"; - private final String westvale_abbey = "Westvale Abbey"; - private final String cradle = "Gaea's Cradle"; - private final String castle_garenbrig = "Castle Garenbrig"; - private final String nyxbloom_ancient = "Nyxbloom Ancient"; - private final String huntmaster = "Huntmaster of the Fells"; - private final String xenagos = "Xenagos, the Reveler"; - private final String thromok = "Thromok the Insatiable"; - private final String world_tree = "The World Tree"; - private final String nylea = "Nylea, Keen-Eyed"; - private final String kogla = "Kogla, the Titan Ape"; - private final String slime = "Acidic Slime"; - private final String druid = "Druid Class"; - - @Test - public void massWarpWorld(){ - removeAllCardsFromLibrary(playerA); - addCard(Zone.BATTLEFIELD, playerA, innkeeper); - addCard(Zone.BATTLEFIELD, playerA, tracker); - addCard(Zone.BATTLEFIELD, playerA, goose); - addCard(Zone.BATTLEFIELD, playerA, galazeth_prismari); - addCard(Zone.BATTLEFIELD, playerA, wood_elves); - addCard(Zone.BATTLEFIELD, playerA, nesting_dragon); - addCard(Zone.BATTLEFIELD, playerA, primeval_titan); - addCard(Zone.BATTLEFIELD, playerA, eternal_witness); - addCard(Zone.BATTLEFIELD, playerA, westvale_abbey); - addCard(Zone.BATTLEFIELD, playerA, cradle); - addCard(Zone.BATTLEFIELD, playerA, castle_garenbrig); - addCard(Zone.BATTLEFIELD, playerA, nyxbloom_ancient); - addCard(Zone.BATTLEFIELD, playerA, huntmaster); - addCard(Zone.BATTLEFIELD, playerA, xenagos); - addCard(Zone.BATTLEFIELD, playerA, thromok); - addCard(Zone.BATTLEFIELD, playerA, world_tree); - addCard(Zone.BATTLEFIELD, playerA, nylea); - addCard(Zone.BATTLEFIELD, playerA, kogla); - addCard(Zone.BATTLEFIELD, playerA, slime); - addCard(Zone.BATTLEFIELD, playerA, druid); - addCard(Zone.BATTLEFIELD, playerA, "Mountain", 10); - addCard(Zone.HAND, playerA, "Warp World"); - castSpell(1, PhaseStep.PRECOMBAT_MAIN, playerA,"Warp World"); - setStopAt(1, PhaseStep.POSTCOMBAT_MAIN); - try { - execute(); - } - catch (ConcurrentModificationException concurrentModificationException){ - concurrentModificationException.printStackTrace(); - Assert.fail("Concurrent Modification Exception raised"); - } - - } -} diff --git a/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityAllEffect.java b/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityAllEffect.java index 36d96f04b6..5f57616512 100644 --- a/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityAllEffect.java +++ b/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityAllEffect.java @@ -102,18 +102,6 @@ public class GainAbilityAllEffect extends ContinuousEffectImpl { perm.addAbility(ability, source.getSourceId(), game); } } - // still as long as the prev. permanent is known to the LKI (e.g. Mikaeus, the Unhallowed) so gained dies triggered ability will trigger - Map LKIBattlefield = game.getLKI().get(Zone.BATTLEFIELD); - if (LKIBattlefield != null) { - for (MageObject mageObject : LKIBattlefield.values()) { - Permanent perm = (Permanent) mageObject; - if (!(excludeSource && perm.getId().equals(source.getSourceId())) && selectedByRuntimeData(perm, source, game)) { - if (filter.match(perm, source.getSourceId(), source.getControllerId(), game)) { - perm.addAbility(ability, source.getSourceId(), game); - } - } - } - } } return true; } diff --git a/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityControlledEffect.java b/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityControlledEffect.java index 1dcc227e8d..388778c935 100644 --- a/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityControlledEffect.java +++ b/Mage/src/main/java/mage/abilities/effects/common/continuous/GainAbilityControlledEffect.java @@ -104,20 +104,6 @@ public class GainAbilityControlledEffect extends ContinuousEffectImpl { } } } - // still as long as the prev. permanent is known to the LKI (e.g. Mikaeus, the Unhallowed) so gained dies triggered ability will trigger - Map LKIBattlefield = game.getLKI().get(Zone.BATTLEFIELD); - if (LKIBattlefield != null) { - for (MageObject mageObject : LKIBattlefield.values()) { - Permanent perm = (Permanent) mageObject; - if (!(excludeSource && perm.getId().equals(source.getSourceId()))) { - if (filter.match(perm, source.getSourceId(), source.getControllerId(), game)) { - for (Ability abilityToAdd : ability) { - perm.addAbility(abilityToAdd, source.getSourceId(), game); - } - } - } - } - } } return true; } diff --git a/Mage/src/main/java/mage/abilities/effects/common/continuous/LoseAbilityAllEffect.java b/Mage/src/main/java/mage/abilities/effects/common/continuous/LoseAbilityAllEffect.java index 076270445a..f6c7595da6 100644 --- a/Mage/src/main/java/mage/abilities/effects/common/continuous/LoseAbilityAllEffect.java +++ b/Mage/src/main/java/mage/abilities/effects/common/continuous/LoseAbilityAllEffect.java @@ -99,18 +99,6 @@ public class LoseAbilityAllEffect extends ContinuousEffectImpl { perm.removeAbilities(ability, source.getSourceId(), game); } } - // still as long as the prev. permanent is known to the LKI (e.g. Mikaeus, the Unhallowed) so gained dies triggered ability will trigger - Map LKIBattlefield = game.getLKI().get(Zone.BATTLEFIELD); - if (LKIBattlefield != null) { - for (MageObject mageObject : LKIBattlefield.values()) { - Permanent perm = (Permanent) mageObject; - if (!(excludeSource && perm.getId().equals(source.getSourceId()))) { - if (filter.match(perm, source.getSourceId(), source.getControllerId(), game)) { - perm.removeAbilities(ability, source.getSourceId(), game); - } - } - } - } } return true; }