[SPIGOT-6516] Fields in NMS packets are final! Created: 11/Jun/21 Updated: 15/Jun/21 Resolved: 11/Jun/21 |
|
Status: | Closed |
Project: | Spigot |
Component/s: | None |
Affects Version/s: | None |
Fix Version/s: | None |
Type: | Bug | Priority: | Major |
Reporter: | retrooper | Assignee: | Unassigned |
Resolution: | Invalid | Votes: | 0 |
Labels: | error, field, modify, nms, packet, reflection, spigot | ||
Environment: |
This is likely one of the first 1.17 builds. I am using Linux Manjaro GNOME. Server is obviously using Java 16 Azul Zulu. |
Version: | This server is running CraftBukkit version 3111-Spigot-66f9d3c-cbf2f67 (MC: 1.17) (Implementing API version 1.17-R0.1-SNAPSHOT) |
Guidelines Read: | Yes |
Description |
In 1.16.5 and all older server versions, fields in NMS packets were NOT final. Which allowed packet listeners (which process incoming and outgoing packets) to modify the data in the packets. Most packet listeners add their netty handlers before the minecraft server. If I modify the data of an incoming packet(server-bound), the vanilla server would process the modified packet. The same applies to outgoing packets, we could modify packets the server was planning to send. This was great, until 1.17. I haven't confirmed that this is the case in all packets, but all the ones I have looked at have had final fields. With reflection we can no longer modify fields. Using unsafe API or some weird java hacks is not really an end solution to this. Spigot can change this for future builds. I understand if you were trying to do some optimizations to the NMS packets, but this is a breaking change for popular plugins like ProtocolLib and my library (packetevents) too.
Edit: Reflection allows you to modify final fields, as long as they are not "static final", but "final" itself works. |
Comments |
Comment by Thibaut Gautier [ 12/Jun/21 ] |
@Black Hole You are officially the moron of yhe year. ProtocolLib relies on reading NMS fields for packet handling. |
Comment by Martoph [ 11/Jun/21 ] |
retrooper Records were introduced to Java 16 as an easy way to store read-only information without the use of getters. |
Comment by Black Hole [ 11/Jun/21 ] |
NMS code is not part of the API. If you need support for packets, please use ProtocolLib. |
Comment by idcidc [ 11/Jun/21 ] |
what.... |
Comment by retrooper [ 11/Jun/21 ] |
What do you mean by records? Is it possible for Spigot to revert this change and make all of them NOT final? What do plugins like ProtocolLib do now? Disable functionality of modifying packets? |
Comment by Parker Hawke [ 11/Jun/21 ] |
This was a change made by Mojang in some of the later snapshots in anticipation of moving to Java 16. In all likelihood they may move packets to records. Spigot does not remove final access unless it intends on modifying a field for CraftBukkit. |