diff --git a/paper-server/src/main/java/io/papermc/paper/util/ItemComponentSanitizer.java b/paper-server/src/main/java/io/papermc/paper/util/ItemComponentSanitizer.java index 35c295183627..f71c2b9a8592 100644 --- a/paper-server/src/main/java/io/papermc/paper/util/ItemComponentSanitizer.java +++ b/paper-server/src/main/java/io/papermc/paper/util/ItemComponentSanitizer.java @@ -62,9 +62,9 @@ DataComponents.CONTAINER, codec(ItemContainerContents.STREAM_CODEC, contents -> * * Any components not present in this map, the prototype value will be sent. */ - public static final Map, UnaryOperator> SANITIZATION_OVERRIDES = Util.make(ImmutableMap., UnaryOperator>builder(), (map) -> { + public static final Map, OverrideConfig> SANITIZATION_OVERRIDES = Util.make(ImmutableMap., OverrideConfig>builder(), (map) -> { put(map, DataComponents.LODESTONE_TRACKER, empty(new LodestoneTracker(Optional.empty(), false))); - put(map, DataComponents.ENCHANTMENTS, empty(dummyEnchantments())); + put(map, DataComponents.ENCHANTMENTS, empty(dummyEnchantments()), true); // ALWAYS OVERRIDE - patched is different put(map, DataComponents.MAX_DAMAGE, empty(1)); put(map, DataComponents.DAMAGE, empty(0)); put(map, DataComponents.CUSTOM_NAME, empty(Component.empty())); @@ -91,8 +91,15 @@ DataComponents.CONTAINER, codec(ItemContainerContents.STREAM_CODEC, contents -> } ).build(); + private record OverrideConfig(UnaryOperator provider, boolean alwaysOverride) { + } + private static void put(ImmutableMap.Builder map, DataComponentType type, UnaryOperator object) { - map.put(type, object); + put(map, type, object, false); + } + + private static void put(ImmutableMap.Builder map, DataComponentType type, UnaryOperator object, boolean alwaysOverride) { + map.put(type, new OverrideConfig<>(object, alwaysOverride)); } private static UnaryOperator empty(T object) { @@ -109,7 +116,7 @@ public static StreamCodec get(DataCompon // Now check the obfuscation, where we lookup if the type is overriden in any of the configurations, if so, wrap the codec GlobalConfiguration.Anticheat.Obfuscation.Items obfuscation = GlobalConfiguration.get().anticheat.obfuscation.items; if (obfuscation.enableItemObfuscation && DataSanitizationUtil.OVERRIDE_TYPES.contains(componentType)) { - return codec(vanillaCodec, new DefaultDataComponentSanitizer<>(componentType, (UnaryOperator) SANITIZATION_OVERRIDES.get(componentType))); + return codec(vanillaCodec, new DefaultDataComponentSanitizer<>(componentType, (OverrideConfig) SANITIZATION_OVERRIDES.get(componentType))); } return vanillaCodec; @@ -192,7 +199,7 @@ private static StreamCodec codec(final StreamCodec delegate, // Serializer that will override the type depending on the asset of the item provided private record DefaultDataComponentSanitizer(DataComponentType type, - @Nullable UnaryOperator override) implements UnaryOperator { + @Nullable OverrideConfig override) implements UnaryOperator { @Override public A apply(final A oldvalue) { @@ -202,9 +209,18 @@ public A apply(final A oldvalue) { if (!DataSanitizationUtil.getAssetObfuscation(targetItemstack).sanitize().contains(this.type)) { return oldvalue; } + // Is we need to force use override, do that + if (this.override != null && this.override.alwaysOverride) { + return this.override.provider.apply(oldvalue); + } - // We don't need to check if its overriden because we KNOW it is if we are serializing it over to the client. - return this.override == null ? (A) scope.itemStack().getPrototype().getOrDefault(this.type, oldvalue) : this.override.apply(oldvalue); + // If we can use the prototype, lets try to do that whenever possible + A value = (A) scope.itemStack().getPrototype().get(this.type); + if (value != null) { + return value; + } else { + return this.override == null ? oldvalue : this.override.provider.apply(oldvalue); + } } }