aboutsummaryrefslogtreecommitdiffhomepage
path: root/patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch
diff options
context:
space:
mode:
Diffstat (limited to 'patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch')
-rw-r--r--patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch86
1 files changed, 86 insertions, 0 deletions
diff --git a/patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch b/patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch
new file mode 100644
index 0000000000..2d07a04363
--- /dev/null
+++ b/patches/server/0717-Prevent-sending-oversized-item-data-in-equipment-and.patch
@@ -0,0 +1,86 @@
+From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
+From: Nassim Jahnke <[email protected]>
+Date: Wed, 1 Dec 2021 12:36:25 +0100
+Subject: [PATCH] Prevent sending oversized item data in equipment and metadata
+
+
+diff --git a/src/main/java/net/minecraft/network/syncher/EntityDataSerializers.java b/src/main/java/net/minecraft/network/syncher/EntityDataSerializers.java
+index 97da8896865ff0bdd4fe8f2155b0830b42051bb1..9ca897d92c5bdd2764d114c74d64c776674d6beb 100644
+--- a/src/main/java/net/minecraft/network/syncher/EntityDataSerializers.java
++++ b/src/main/java/net/minecraft/network/syncher/EntityDataSerializers.java
+@@ -42,7 +42,7 @@ public class EntityDataSerializers {
+ public static final EntityDataSerializer<ItemStack> ITEM_STACK = new EntityDataSerializer<ItemStack>() {
+ @Override
+ public void write(FriendlyByteBuf buf, ItemStack value) {
+- buf.writeItem(value);
++ buf.writeItem(net.minecraft.world.entity.LivingEntity.sanitizeItemStack(value, true)); // Paper - prevent oversized data
+ }
+
+ @Override
+diff --git a/src/main/java/net/minecraft/server/level/ServerEntity.java b/src/main/java/net/minecraft/server/level/ServerEntity.java
+index e0802f1cb73a80b08482832c2b269ac8485d5c1a..8d2870c780c4c253f6570c7ef73f6e7c2ccc46ad 100644
+--- a/src/main/java/net/minecraft/server/level/ServerEntity.java
++++ b/src/main/java/net/minecraft/server/level/ServerEntity.java
+@@ -335,7 +335,10 @@ public class ServerEntity {
+ ItemStack itemstack = ((LivingEntity) this.entity).getItemBySlot(enumitemslot);
+
+ if (!itemstack.isEmpty()) {
+- list.add(Pair.of(enumitemslot, itemstack.copy()));
++ // Paper start - prevent oversized data
++ final ItemStack sanitized = LivingEntity.sanitizeItemStack(itemstack.copy(), false);
++ list.add(Pair.of(enumitemslot, sanitized));
++ // Paper end
+ }
+ }
+
+diff --git a/src/main/java/net/minecraft/world/entity/LivingEntity.java b/src/main/java/net/minecraft/world/entity/LivingEntity.java
+index 1e0dfa8e04e02e552d9233c12be915778bb6e09c..c2d06bd94c829e1e9cf00c20e5e2547c60d4b0ff 100644
+--- a/src/main/java/net/minecraft/world/entity/LivingEntity.java
++++ b/src/main/java/net/minecraft/world/entity/LivingEntity.java
+@@ -3191,7 +3191,10 @@ public abstract class LivingEntity extends Entity implements Attackable {
+ equipmentChanges.forEach((enumitemslot, itemstack) -> {
+ ItemStack itemstack1 = itemstack.copy();
+
+- list.add(Pair.of(enumitemslot, itemstack1));
++ // Paper start - prevent oversized data
++ ItemStack toSend = sanitizeItemStack(itemstack1, true);
++ list.add(Pair.of(enumitemslot, toSend));
++ // Paper end
+ switch (enumitemslot.getType()) {
+ case HAND:
+ this.setLastHandItem(enumitemslot, itemstack1);
+@@ -3204,6 +3207,34 @@ public abstract class LivingEntity extends Entity implements Attackable {
+ ((ServerLevel) this.level()).getChunkSource().broadcast(this, new ClientboundSetEquipmentPacket(this.getId(), list));
+ }
+
++ // Paper start - prevent oversized data
++ public static ItemStack sanitizeItemStack(final ItemStack itemStack, final boolean copyItemStack) {
++ if (itemStack.isEmpty() || !itemStack.hasTag()) {
++ return itemStack;
++ }
++
++ final ItemStack copy = copyItemStack ? itemStack.copy() : itemStack;
++ final CompoundTag tag = copy.getTag();
++ if (copy.is(Items.BUNDLE) && tag.get("Items") instanceof ListTag oldItems && !oldItems.isEmpty()) {
++ // Bundles change their texture based on their fullness.
++ org.bukkit.inventory.meta.BundleMeta bundleMeta = (org.bukkit.inventory.meta.BundleMeta) copy.asBukkitMirror().getItemMeta();
++ int sizeUsed = 0;
++ for (org.bukkit.inventory.ItemStack item : bundleMeta.getItems()) {
++ int scale = 64 / item.getMaxStackSize();
++ sizeUsed += scale * item.getAmount();
++ }
++ // Now we add a single fake item that uses the same amount of slots as all other items.
++ ListTag items = new ListTag();
++ items.add(new ItemStack(Items.PAPER, sizeUsed).save(new CompoundTag()));
++ tag.put("Items", items);
++ }
++ if (tag.get("BlockEntityTag") instanceof CompoundTag blockEntityTag) {
++ blockEntityTag.remove("Items");
++ }
++ return copy;
++ }
++ // Paper end
++
+ private ItemStack getLastArmorItem(EquipmentSlot slot) {
+ return (ItemStack) this.lastArmorItemStacks.get(slot.getIndex());
+ }