2024-05-20 16:20:47 +02:00
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
From: Jake Potrebic <jake.m.potrebic@gmail.com>
Date: Sun, 12 May 2024 15:49:36 -0700
Subject: [PATCH] Fix issues with Recipe API
diff --git a/src/main/java/net/minecraft/world/item/crafting/ShapedRecipe.java b/src/main/java/net/minecraft/world/item/crafting/ShapedRecipe.java
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
--- a/src/main/java/net/minecraft/world/item/crafting/ShapedRecipe.java
+++ b/src/main/java/net/minecraft/world/item/crafting/ShapedRecipe.java
2024-10-24 17:16:31 +02:00
@@ -0,0 +0,0 @@ public class ShapedRecipe implements CraftingRecipe {
2024-05-20 16:20:47 +02:00
char c = 'a';
2024-10-24 17:16:31 +02:00
for (Optional<Ingredient> list : this.pattern.ingredients()) {
2024-05-20 16:20:47 +02:00
RecipeChoice choice = CraftRecipe.toBukkit(list);
- if (choice != null) {
+ if (choice != RecipeChoice.empty()) { // Paper
recipe.setIngredient(c, choice);
}
diff --git a/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java b/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
--- a/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
+++ b/src/main/java/org/bukkit/craftbukkit/inventory/CraftRecipe.java
2024-11-01 11:30:40 +01:00
@@ -0,0 +0,0 @@ public interface CraftRecipe extends Recipe {
void addToCraftingManager();
default Optional<Ingredient> toNMSOptional(RecipeChoice bukkit, boolean requireNotEmpty) {
- return (bukkit == null) ? Optional.empty() : Optional.of(this.toNMS(bukkit, requireNotEmpty));
+ return (bukkit == null || bukkit == RecipeChoice.empty()) ? Optional.empty() : Optional.of(this.toNMS(bukkit, requireNotEmpty)); // Paper - support "empty" choices
}
default Ingredient toNMS(RecipeChoice bukkit, boolean requireNotEmpty) {
2024-05-20 16:20:47 +02:00
@@ -0,0 +0,0 @@ public interface CraftRecipe extends Recipe {
2024-10-24 17:16:31 +02:00
stack = Ingredient.of(((RecipeChoice.MaterialChoice) bukkit).getChoices().stream().map((mat) -> CraftItemType.bukkitToMinecraft(mat)));
2024-05-20 16:20:47 +02:00
} else if (bukkit instanceof RecipeChoice.ExactChoice) {
2024-10-24 19:29:35 +02:00
stack = Ingredient.ofStacks(((RecipeChoice.ExactChoice) bukkit).getChoices().stream().map((mat) -> CraftItemStack.asNMSCopy(mat)).toList());
2024-11-01 11:30:40 +01:00
+ // Paper start - support "empty" choices - legacy method that spigot might incorrectly call
+ // Their impl of Ingredient.of() will error, ingredients need at least one entry.
+ // Callers running into this exception may have passed an incorrect empty() recipe choice to a non-empty slot or
+ // spigot calls this method in a wrong place.
2024-05-20 16:20:47 +02:00
+ } else if (bukkit == RecipeChoice.empty()) {
2024-11-01 11:30:40 +01:00
+ throw new IllegalArgumentException("This ingredient cannot be empty");
+ // Paper end - support "empty" choices
2024-05-20 16:20:47 +02:00
} else {
throw new IllegalArgumentException("Unknown recipe stack instance " + bukkit);
}
@@ -0,0 +0,0 @@ public interface CraftRecipe extends Recipe {
2024-10-25 22:33:37 +02:00
}
public static RecipeChoice toBukkit(Optional<Ingredient> list) {
- return list.map(CraftRecipe::toBukkit).orElse(null);
+ return list.map(CraftRecipe::toBukkit).orElse(RecipeChoice.empty()); // Paper - fix issue with recipe API
}
public static RecipeChoice toBukkit(Ingredient list) {
2024-12-03 20:34:55 +01:00
if (list.isEmpty()) {
2024-05-20 16:20:47 +02:00
- return null;
+ return RecipeChoice.empty(); // Paper - null breaks API contracts
}
2024-10-24 17:16:31 +02:00
if (list.isExact()) {
2024-05-20 16:20:47 +02:00
diff --git a/src/test/java/io/papermc/paper/inventory/recipe/TestRecipeChoice.java b/src/test/java/io/papermc/paper/inventory/recipe/TestRecipeChoice.java
new file mode 100644
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000
--- /dev/null
+++ b/src/test/java/io/papermc/paper/inventory/recipe/TestRecipeChoice.java
@@ -0,0 +0,0 @@
+package io.papermc.paper.inventory.recipe;
+
+import java.util.Iterator;
+import org.bukkit.Bukkit;
+import org.bukkit.inventory.Recipe;
2024-10-21 00:06:54 +02:00
+import org.bukkit.support.environment.AllFeatures;
2024-05-20 16:20:47 +02:00
+import org.junit.jupiter.api.Test;
+
+import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
2024-10-21 00:06:54 +02:00
+@AllFeatures
+class TestRecipeChoice {
2024-05-20 16:20:47 +02:00
+
+ @Test
+ void testRecipeChoices() {
+ final Iterator<Recipe> iter = Bukkit.recipeIterator();
+ boolean foundRecipes = false;
+ while (iter.hasNext()) {
+ foundRecipes = true;
+ assertDoesNotThrow(iter::next, "Failed to convert a recipe to Bukkit recipe!");
+ }
+ assertTrue(foundRecipes, "No recipes found!");
+ }
+}
2024-10-21 00:06:54 +02:00
diff --git a/src/test/java/org/bukkit/support/DummyServerHelper.java b/src/test/java/org/bukkit/support/DummyServerHelper.java
2024-05-20 16:20:47 +02:00
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
2024-10-21 00:06:54 +02:00
--- a/src/test/java/org/bukkit/support/DummyServerHelper.java
+++ b/src/test/java/org/bukkit/support/DummyServerHelper.java
@@ -0,0 +0,0 @@ public final class DummyServerHelper {
// Paper end - testing additions
2024-05-20 16:20:47 +02:00
2024-10-21 00:06:54 +02:00
io.papermc.paper.configuration.GlobalConfigTestingBase.setupGlobalConfigForTest(); // Paper - configuration files - setup global configuration test base
2024-05-20 16:20:47 +02:00
+
2024-10-21 00:06:54 +02:00
+ // Paper start - add test for recipe conversion
+ when(instance.recipeIterator()).thenAnswer(ignored ->
+ com.google.common.collect.Iterators.transform(
2024-10-25 19:15:40 +02:00
+ RegistryHelper.getDataPack().getRecipeManager().recipes.byType.entries().iterator(),
2024-10-21 00:06:54 +02:00
+ input -> input.getValue().toBukkitRecipe()
+ )
+ );
+ // Paper end - add test for recipe conversion
return instance;
}
}