TerraFirmaCraft

TerraFirmaCraft

3M Downloads

`TFCAnimalProperties.isFood` is shadowing (vanilla) `Animal.isFood`

JaisDK opened this issue ยท 3 comments

commented

Hi,

I have an addon for TFC that crashes with a NoSuchMethodError on some systems. Are there any situations where the method isFood is not available? It's not crashing in my dev or test env.

Crash log: https://mclo.gs/A54O1Nu

Here's the classes calling isFood https://github.com/JDKDigital/tfcgroomer/blob/dev-1.20.1/src/main/java/cy/jdkdigital/tfcgroomer/common/block/entity/GroomingStationBlockEntity.java#L80

https://github.com/TerraFirmaCraft/TerraFirmaCraft/blob/1.20.x/src/main/java/net/dries007/tfc/common/entities/livestock/TFCAnimalProperties.java#L604

commented

This is an unfortunate problem where compilation with obfuscation is, seemingly, nondeterministic. What's happening here is in TFC, TFCAnimalProperties.isFood() is shadowing the vanilla method Animal.isFood, and depending on how that compiles (in TFC), it can cause weird situations. As it stands, in the released version of 3.2.13, it compiles to the following:

// TFCAnimal.class
public boolean m_6898_(ItemStack stack) {
    return super.m_6898_(stack);
}

// TFCAnimalProperties.class
default boolean m_6898_(ItemStack stack) {
    return (this.eatsRottenFood() || !FoodCapability.isRotten(stack)) && Helpers.isItem(stack, this.getFoodTag());
}

This is, a problem. It's unfortunately a problem that we can't solve without breaking most addons that interact here, because it would involve a rename of isFood to something like isFoodTFC to avoid the shadowing. It's also a problem that is gone in 1.21 (as the lack of obfuscation at dev time means that the method never has this uncertain, non-deterministic renaming.

If your addon works, at the current time, will entirely depend on what it is being compiled as - yes, the method doesn't exist at runtime because it's being shadowed. Forge will, seemingly, correctly obfuscate references to TFCAnimal::isFood but not TFCAnimalProperties.isFood. You might be able to work around this, although it is a ticking time bomb in our code as it stands.

commented

Would it be safe to copy the contents of isFood and use the methods in there?

    public static void tickServer(Level level, BlockPos pos, BlockState state, GroomingStationBlockEntity e) {
        if (e.counter-- <= 0 && level instanceof ServerLevel serverLevel) {
            e.counter = 1200;

            List<ItemStack> stacks = new ArrayList<>();
            for (int i = 0; i < e.inventory.getSlots(); i++) {
                var stack = e.inventory.getStackInSlot(i);
                if (!stack.isEmpty()) {
                    stacks.add(stack);
                }
            }
            List<Animal> entities = level.getEntitiesOfClass(Animal.class, (new AABB(pos).inflate(e.range, 1d, e.range))).stream().toList();
            if (!entities.isEmpty()) {
                Player fakePlayer = FakePlayerFactory.get(serverLevel, new GameProfile(PLAYER_UUID, "grooming_station"));
                entities.forEach(animal -> {
                    if (animal instanceof TFCAnimalProperties tfcAnimal) {
                        for (ItemStack stack : stacks) {
                            if (!stack.isEmpty() && tfcAnimal.isHungry() && isFood(tfcAnimal, stack)) {
                                tfcAnimal.eatFood(stack, InteractionHand.MAIN_HAND, fakePlayer);
                                break;
                            }
                        }
                    }
                });
            }
        }
    }

    static boolean isFood(TFCAnimalProperties tfcAnimal, ItemStack stack) {
        return (tfcAnimal.eatsRottenFood() || !FoodCapability.isRotten(stack)) && Helpers.isItem(stack, tfcAnimal.getFoodTag());
    }
commented

Yes, it's safe to copy the logic.