[SPIGOT-2991] Spigot filters invalid enchants from being sent to the client Created: 05/Jan/17 Updated: 07/May/19 Resolved: 05/Jan/17 |
|
Status: | Resolved |
Project: | Spigot |
Component/s: | None |
Affects Version/s: | None |
Fix Version/s: | None |
Type: | Bug | Priority: | Minor |
Reporter: | Vlad Ardelean | Assignee: | Unassigned |
Resolution: | Won't Fix | Votes: | 0 |
Labels: | 1.11, enchantment, spigot |
Issue Links: |
|
Description |
I've only tested this on 1.11+ versions, as that's the only versions where this is relevant. As you might already know, mojang chose to change the way their clients consider an item "enchanted", thus adding the enchantment glow. Before 1.11, any items having the "ench" tag, empty or not, would have the enchantment glint on them. After 1.11 this behavior was changed: Now items not only have to have the "ench" tag, but that list also has to contain some elements, valid or not it doesn't matter. While this works just nice in vanilla (ex. command: /give @p Dirt 1 0 {ench:[\{lvl:-1s, id:-1s}]} - will give you an enchanted dirt block without actual enchantments), doing the same thing while using the Spigot server won't work. My guess is that there's some sort of invalid enchantment filter in place which blocks invalid enchants from being sent to the client, but I really can't see the use of such thing, as it's blocking many nice applications that could be done this way. Also, I'm sure it doesn't block any kinds of "custom" tags you might add to the item. Using a slightly modified version of the official 1.11 client compiled from the code provided by MCP, I added a debug message for each item in a player's inventory displaying it's nbt tag compound, and it did actually display any custom tags I tried adding to items, so I think this "filter" only works for enchantments. If you could fix this annoying bug, you would certainly improve the experience for both plugin developers and map-makers who use spigot as their server of choice. |
Comments |
Comment by Vlad Ardelean [ 09/Jan/17 ] |
Ok, I understand, but even directly sending a packet to the client, for example, the setslot packet, won't work. That's why I thought there's some filter in place. I understand why the API isn't allowing it, but to my understanding, just sending a packet shouldn't have anything to do with the API. However, the item will still just show up without the invalid tags. |
Comment by md_5 [ 09/Jan/17 ] |
Yes, the API isn't meant to break across versions. |
Comment by Vlad Ardelean [ 09/Jan/17 ] |
OK then, but even if Mojang removes that functionality, would it be so hard to remove that part of the API? |
Comment by md_5 [ 08/Jan/17 ] |
Mods don't represent Mojang. I'm a mod on their tracker also.. |
Comment by Vlad Ardelean [ 08/Jan/17 ] |
Screenshot with a quote by a mod: http://i.imgur.com/dgCd827.png >And then how do you remove these enchants if they aren't ever returned by the API? I guess you could just remove them using NMS. Remember they are not actual enchants, and therefore won't influence anything (besides the enchantment glint) even if never removed. >Since you're already hacking into the API with reflection, why not just hack further and register them properly in the server as enchants so they actually work. You've incorrectly used reflection to hack around the API and then complaining it doesn't work because you didn't do it properly or fully. I managed to do that already. However, but I was just looking for a way to allow invalid enchantments (via commands if possible - eg: the /give command). >It's not that simple, the NBT tag isn't ever stored. It is fully deserialized to Bukkit and then later re-serialized, which is why you need to actually register them to the server (see next point). I understand, and I also see that it's not up to me how this is implemented, I just made a suggestion. |
Comment by md_5 [ 08/Jan/17 ] |
>The way they recommended me to do that currently is by using invalid enchants >Also, why would it be hard to edit the API so it will only return the valid enchantments on the item? You should just check if their ids are registered, and if not, simply don't return them, but leave them in the nbt tag. >You should just check if their ids are registered, and if not, simply don't return them, but leave them in the nbt tag. > I used reflection to set the acceptingNew field to true The Bukkit API will not support custom blocks/entities/items/enchants/whatever because it is a plugin, not modding API. |
Comment by Vlad Ardelean [ 08/Jan/17 ] |
For the first part, I gotta admit I used reflection to set the acceptingNew field to true. As for mojang stopping to accept invalid enchants, I highly doubt it. I mean, I've reported the 1.11 change (the one that made empty ench:[] tags still not display the item glow for clients) and they said it was an intended change. The way they recommended me to do that currently is by using invalid enchants, and that is currently the only way to add glow to unenchanted items, therefore I highly doubt they will remove it. Also, why would it be hard to edit the API so it will only return the valid enchantments on the item? You should just check if their ids are registered, and if not, simply don't return them, but leave them in the nbt tag. |
Comment by md_5 [ 06/Jan/17 ] |
1) java.lang.IllegalStateException: No longer accepting new enchantments (can only be done by the server implementation) at org.bukkit.enchantments.Enchantment.registerEnchantment(Enchantment.java:287) ~[classes/:?] at net.md_5.TestPlugin.onEnable(TestPlugin.java:32) ~[?:?] at org.bukkit.plugin.java.JavaPlugin.setEnabled(JavaPlugin.java:271) ~[classes/:?] at org.bukkit.plugin.java.JavaPluginLoader.enablePlugin(JavaPluginLoader.java:337) [classes/:?] at org.bukkit.plugin.SimplePluginManager.enablePlugin(SimplePluginManager.java:405) [classes/:?] at org.bukkit.craftbukkit.CraftServer.enablePlugin(CraftServer.java:376) [classes/:?] at org.bukkit.craftbukkit.CraftServer.enablePlugins(CraftServer.java:326) [classes/:?] at net.minecraft.server.MinecraftServer.t(MinecraftServer.java:419) [classes/:?] at net.minecraft.server.MinecraftServer.l(MinecraftServer.java:380) [classes/:?] at net.minecraft.server.MinecraftServer.a(MinecraftServer.java:335) [classes/:?] at net.minecraft.server.DedicatedServer.init(DedicatedServer.java:272) [classes/:?] at net.minecraft.server.MinecraftServer.run(MinecraftServer.java:542) [classes/:?] at java.lang.Thread.run(Thread.java:745) [?:1.8.0_111] java.lang.IllegalStateException: No longer accepting new enchantments (can only be done by the server implementation)
I don't even know how you're using that method. |
Comment by Vlad Ardelean [ 06/Jan/17 ] |
1) What is that method for than? Or why is it exposed as a public method to other plugins? Actually, the JavaDocs say it's for no purpose. Anyways, let's be honest: It's a really useful method and it's used by many, especially plugins who try to implement "custom" enchantments features tied in as much as possible to vanilla enchantments. 2) I don't really understand how am I really "special casing the API". Can you detail please? 3) The ability for adding custom enchants? I really can't understand why would that happen, and how it could possibly be considered a bug... 4) Well the vanilla server does seem to store invalid enchants, and I don't see why Bukkit wouldn't do the same. It already saves custom NBT data on items, so why wouldn't it also save invalid enchant data? - It's also in it's NBT. I understand that Bukkit is NOT a modding platform in the way many look at it, like Forge, for instance, which allows addition of custom blocks, entities, etc. However, Bukkit supports changing the behavior of anything that is already in the game. Adding a custom enchantment for only server-side use, which plugins can later read and use (rather than using stupid improvisations, like writing information in the item's lore or using custom NBT tags), won't require new assets for the client (models, sounds, textures, etc. - although this can also be done nowadays through the use of resource packs), but could at the same time extend the functionality of vanilla enchantments in a mod-like way. |
Comment by md_5 [ 06/Jan/17 ] |
1) not aware of any supported method to register custom enchants, what method are you using? The javadocs make it clear that the registerEnchantment method is not for this purpose. |
Comment by Vlad Ardelean [ 06/Jan/17 ] |
Bukkit does not support custom enchants. There is no way to obtain the stack you are describing via the Bukkit API. I am aware of that. However, you could obtain it using the /give command or NMS code. Also, the first part is not entirely true. The enchantment class exposes a public method which allows you to register custom enchants. Allowing such a stack to exist would break plugins which depend on only valid API. I honestly can't see how that might happen. If the plugins depend only on the Bukkit API, then any method they might use, for instance, the getEnchantments method on an ItemStack, should only return the valid enchantments present on the item, ignoring the invalid ones. It is actually more likely for the current way the API works to break such plugins. Allowing the registration of custom enchantments (don't get me wrong, that's a nice feature and I'm sure I'm not the only one who uses it) through the API might have plugins that expect receiving a list of only vanilla enchants after using getEnchantments actually get a list including one or more custom enchants, which they might not know how to handle. Therefore, in my opinion, I feel like removing that invalid enchantment filter would not only allow both plugin developers and map-makers to create glowing items (for map-makers that's the only way currently, as they can't register custom enchantments through the API and use the "temporary" fix described by me in my first comment), but wouldn't affect other plugins only depending on the Bukkit API. |
Comment by md_5 [ 05/Jan/17 ] |
Bukkit does not support custom enchants. There is no way to obtain the stack you are describing via the Bukkit API. |
Comment by Vlad Ardelean [ 05/Jan/17 ] |
Because players will be able to see that enchantment, unless I add the HideEnchants flag to the item. In that case, however, players would also not see the enchants they might try to add to the item later, so it's definitely something that wouldn't work. |
Comment by Black Hole [ 05/Jan/17 ] |
Why don't you use an existing enchantment that has no use for the item? Like "Fire Protection" for weapons. |