Difference between revisions of "Server List Ping"
Dreamscached (talk | contribs) m (Add space to title) |
m (This was tested and found out through testing with the server.properties file of a vanilla server.) |
||
(38 intermediate revisions by 19 users not shown) | |||
Line 1: | Line 1: | ||
− | '''Server List Ping''' ('''SLP''') is an interface provided by Minecraft servers which supports querying the MOTD, player count, max players and server version via the usual port. SLP is part of the [[Protocol]], but it can be disabled. The Notchian client uses this interface to display the multiplayer server list, hence the name. The SLP process changed in {{Minecraft Wiki|1.7}} in a non-backwards compatible way, but | + | '''Server List Ping''' ('''SLP''') is an interface provided by Minecraft servers which supports querying the MOTD, player count, max players and server version via the usual port. SLP is part of the [[Protocol]], but it can be disabled. The Notchian client uses this interface to display the multiplayer server list, hence the name. The SLP process changed in {{Minecraft Wiki|Java Edition 1.7}} in a non-backwards compatible way, but modern clients and servers still support both the [[#Current|new]] and [[#1.6|old]] process. |
== Current (1.7+) == | == Current (1.7+) == | ||
Line 17: | Line 17: | ||
|rowspan="4"| 0x00 | |rowspan="4"| 0x00 | ||
| Protocol Version | | Protocol Version | ||
− | | VarInt | + | | {{Type|VarInt}} |
| See [[protocol version numbers]]. The version that the client plans on using to connect to the server (which is not important for the ping). If the client is pinging to determine what version to use, by convention <code>-1</code> should be set. | | See [[protocol version numbers]]. The version that the client plans on using to connect to the server (which is not important for the ping). If the client is pinging to determine what version to use, by convention <code>-1</code> should be set. | ||
+ | {{Warning|Setting invalid (nonexistent) version as the protocol version <i>might</i> cause some servers to close connection after this packet}}<br/> | ||
+ | {{Warning|See [[Protocol version numbers]] for a list of valid protocol versions.}} | ||
|- | |- | ||
| Server Address | | Server Address | ||
− | | String | + | | {{Type|String}} |
| Hostname or IP, e.g. localhost or 127.0.0.1, that was used to connect. The Notchian server does not use this information. Note that SRV records are a complete redirect, e.g. if _minecraft._tcp.example.com points to mc.example.org, users connecting to example.com will provide mc.example.org as server address in addition to connecting to it. | | Hostname or IP, e.g. localhost or 127.0.0.1, that was used to connect. The Notchian server does not use this information. Note that SRV records are a complete redirect, e.g. if _minecraft._tcp.example.com points to mc.example.org, users connecting to example.com will provide mc.example.org as server address in addition to connecting to it. | ||
|- | |- | ||
| Server Port | | Server Port | ||
− | | Unsigned Short | + | | {{Type|Unsigned Short}} |
| Default is 25565. The Notchian server does not use this information. | | Default is 25565. The Notchian server does not use this information. | ||
|- | |- | ||
| Next state | | Next state | ||
− | | VarInt | + | | {{Type|VarInt}} |
| Should be 1 for [[Protocol#Status|status]], but could also be 2 for [[Protocol#Login|login]]. | | Should be 1 for [[Protocol#Status|status]], but could also be 2 for [[Protocol#Login|login]]. | ||
|} | |} | ||
Line 36: | Line 38: | ||
The client follows up with a [[Protocol#Status Request|Status Request]] packet. This packet has no fields. | The client follows up with a [[Protocol#Status Request|Status Request]] packet. This packet has no fields. | ||
+ | The client is also able to skip this part entirely and send a [[Protocol#Ping Request|Ping Request]] instead. | ||
{| class="wikitable" | {| class="wikitable" | ||
Line 59: | Line 62: | ||
| 0x00 | | 0x00 | ||
| JSON Response | | JSON Response | ||
− | | String | + | | {{Type|String}} |
− | | See below; as with all strings this is prefixed by its length as a VarInt( | + | | See below; as with all strings this is prefixed by its length as a VarInt(3-byte max) |
|} | |} | ||
The JSON Response field is a [[wikipedia:JSON|JSON]] object which has the following format: | The JSON Response field is a [[wikipedia:JSON|JSON]] object which has the following format: | ||
− | <syntaxhighlight lang=" | + | <syntaxhighlight lang="json"> |
{ | { | ||
"version": { | "version": { | ||
− | "name": "1.19", | + | "name": "1.19.4", |
− | "protocol": | + | "protocol": 762 |
}, | }, | ||
"players": { | "players": { | ||
Line 82: | Line 85: | ||
}, | }, | ||
"description": { | "description": { | ||
− | "text": "Hello world" | + | "text": "Hello, world!" |
}, | }, | ||
"favicon": "data:image/png;base64,<data>", | "favicon": "data:image/png;base64,<data>", | ||
− | " | + | "enforcesSecureChat": false |
} | } | ||
</syntaxhighlight> | </syntaxhighlight> | ||
− | + | The ''name'' field of the ''version'' object should be considered mandatory. On newer Notchian client versions (1.20+?), it may be omitted and will be treated as though it were the string "Old" if so. | |
− | The ''description'' field is a [[ | + | The ''description'' field is optional. If specified, it should be a [[Text formatting#Text components|text component]]. Note that the Notchian server has no way of providing actual text component data; instead section sign-based codes are embedded within the text of the object. However, third-party servers such as Spigot and Paper will return full components, so make sure you can handle both. |
− | The ''favicon'' field is optional. | + | The ''favicon'' field is optional. If specified, it should be a [[wikipedia:Portable Network Graphics|PNG]] image that is [[wikipedia:Base64|Base64]] encoded (without newlines: <code>\n</code>, new lines no longer work since 1.13) and prepended with <code>data:image/png;base64,</code>. It should also be noted that the source image must be exactly 64x64 pixels, otherwise the Notchian client will not render the image. |
− | The '' | + | The ''players'' field is optional. If omitted, "???" will be displayed in dark grey in place of the player count. |
− | + | The numbers ''max'' and ''online'' in the players field have a maximum supported value of 2<sup>31</sup>-1 (2,147,483,647) by the vanilla Minecraft server. Larger values were observed to be set to the default value of 20. | |
+ | |||
+ | If the ''sample'' field of the ''players'' object is present, the player names (but not their UUIDs) will be shown in order in the tooltip when hovering over the player count (or version name if incompatible). If empty, missing, or the wrong type, no tooltip will appear. Notchian servers will omit this field if there are no players on the server or if <code>hide-online-players</code> is false. There doesn't appear to be a hard limit on how many players may be specified. | ||
+ | |||
+ | If a client [[Protocol#Client_Information_.28configuration.29|has "Allow Server Listings" set to "OFF"]], Notchian servers will report their name as "Anonymous Player" and UUID as "00000000-0000-0000-0000-000000000000". This is not reliable; making a request before the player has fully joined may cause their choice to not be respected. | ||
+ | |||
+ | On most Notchian client versions, UUIDs are mandatory and must be well formed even though it is unused by Notchian clients. Newer versions (1.20+?) will tolerate malformed or missing UUIDs and act as though ''sample'' was omitted if so. | ||
+ | |||
+ | If the server has No Chat Reports installed and hasn't disabled this feature, an additional field ''preventsChatReports'' will be added to the end with a value of true causing clients with No Chat Reports to display an icon indicating the server is a "Safe Server". | ||
After receiving the Response packet, the client may send the next packet to help calculate the server's latency, or if it is only interested in the above information it can disconnect here. | After receiving the Response packet, the client may send the next packet to help calculate the server's latency, or if it is only interested in the above information it can disconnect here. | ||
− | If the client does not receive a properly formatted response, | + | If the client does not receive a properly formatted response (e.g. a field is missing or the wrong type), it will close the socket and instead attempt a [[Server_List_Ping#1.6|legacy ping]]. |
=== Ping Request === | === Ping Request === | ||
Line 115: | Line 126: | ||
| 0x01 | | 0x01 | ||
| Payload | | Payload | ||
− | | Long | + | | {{Type|Long}} |
| May be any number. Notchian clients use a system-dependent time value which is counted in milliseconds. | | May be any number. Notchian clients use a system-dependent time value which is counted in milliseconds. | ||
|} | |} | ||
Line 131: | Line 142: | ||
| 0x01 | | 0x01 | ||
| Payload | | Payload | ||
− | | Long | + | | {{Type|Long}} |
| Should be the same as sent by the client | | Should be the same as sent by the client | ||
|} | |} | ||
Line 143: | Line 154: | ||
=== Examples === | === Examples === | ||
− | * [https://gist.github.com/csh/2480d14fbbb33b4bbae3 C#] | + | * [https://gist.github.com/csh/2480d14fbbb33b4bbae3 C#], [https://github.com/TheVeryStarlk/Pingo C#] |
* [https://gist.github.com/zh32/7190955 Java] | * [https://gist.github.com/zh32/7190955 Java] | ||
* [https://gist.github.com/1209061 Python] | * [https://gist.github.com/1209061 Python] | ||
+ | * [https://github.com/Duckulus/mc-honeypot Rust (Server)] | ||
+ | * [https://github.com/Sch8ill/MCClient-lib MCClient-lib (Python3)] | ||
* [https://gist.github.com/ewized/97814f57ac85af7128bf Python3] | * [https://gist.github.com/ewized/97814f57ac85af7128bf Python3] | ||
* [https://github.com/xPaw/PHP-Minecraft-Query PHP] | * [https://github.com/xPaw/PHP-Minecraft-Query PHP] | ||
Line 151: | Line 164: | ||
* [https://gitlab.bixilon.de/bixilon/minosoft/-/blob/master/src/main/java/de/bixilon/minosoft/protocol/protocol/LANServerListener.kt LAN Server Listener (Kotlin)] | * [https://gitlab.bixilon.de/bixilon/minosoft/-/blob/master/src/main/java/de/bixilon/minosoft/protocol/protocol/LANServerListener.kt LAN Server Listener (Kotlin)] | ||
* [https://github.com/stackotter/delta-client/blob/main/Sources/Core/Sources/Network/LANServerEnumerator.swift Swift] | * [https://github.com/stackotter/delta-client/blob/main/Sources/Core/Sources/Network/LANServerEnumerator.swift Swift] | ||
− | * [https://github.com/ | + | * [https://github.com/dreamscached/minequery Go] |
+ | * [https://github.com/Sch8ill/mclib Go] | ||
* [https://github.com/PauldeKoning/minecraft-server-handshake Node.js] | * [https://github.com/PauldeKoning/minecraft-server-handshake Node.js] | ||
+ | * [https://github.com/LhAlant/MinecraftSLP/ C] | ||
+ | * [https://github.com/mcstatus-io/mcutil/ Go] | ||
== 1.6 == | == 1.6 == | ||
Line 192: | Line 208: | ||
The remainder is null character (that is <code>00 00</code>) delimited fields: | The remainder is null character (that is <code>00 00</code>) delimited fields: | ||
− | # Protocol version (e.g. <code> | + | # Protocol version (e.g. <code>47</code>) |
− | # Minecraft server version (e.g. <code>1. | + | # Minecraft server version (e.g. <code>1.4.2</code>) |
# Message of the day (e.g. <code>A Minecraft Server</code>) | # Message of the day (e.g. <code>A Minecraft Server</code>) | ||
# Current player count | # Current player count | ||
Line 213: | Line 229: | ||
* [https://gist.github.com/6281388 Ruby] | * [https://gist.github.com/6281388 Ruby] | ||
* [https://github.com/winny-/mcstat PHP] | * [https://github.com/winny-/mcstat PHP] | ||
− | * [https://github.com/ | + | * [https://github.com/Duckulus/mc-honeypot/blob/1514807e8af7f7cbfd13111fec334b9f4883b605/src/server/legacy.rs Rust (Server)] |
+ | * [https://github.com/dreamscached/minequery Go] | ||
+ | * [https://github.com/mcstatus-io/mcutil Go] | ||
== 1.4 to 1.5 == | == 1.4 to 1.5 == | ||
Line 221: | Line 239: | ||
=== Examples === | === Examples === | ||
+ | * [https://github.com/Sch8ill/MCClient-lib Python3 (MCClient-lib)] | ||
* [https://gist.github.com/5795159 PHP] | * [https://gist.github.com/5795159 PHP] | ||
* [https://gist.github.com/4574114 Java] | * [https://gist.github.com/4574114 Java] | ||
* [https://gist.github.com/6223787 C#] | * [https://gist.github.com/6223787 C#] | ||
− | * [https://github.com/ | + | * [https://github.com/dreamscached/minequery Go] |
+ | * [https://github.com/mcstatus-io/mcutil Go] | ||
== Beta 1.8 to 1.3 == | == Beta 1.8 to 1.3 == | ||
Line 237: | Line 257: | ||
|- | |- | ||
| Packet Id | | Packet Id | ||
− | | Byte | + | | {{Type|Byte}} |
| The kick packet id: <code>0xFF</code> | | The kick packet id: <code>0xFF</code> | ||
|- | |- | ||
| Packet Length | | Packet Length | ||
− | | Short | + | | {{Type|Short}} |
− | | The | + | | The length of the following string in characters (NOT BYTES). Max is 256. |
|- | |- | ||
| MOTD | | MOTD | ||
|rowspan="3"| A [http://en.wikipedia.org/wiki/UTF-16 UTF-16BE] encoded string | |rowspan="3"| A [http://en.wikipedia.org/wiki/UTF-16 UTF-16BE] encoded string | ||
− | | From here on out, all fields should be | + | | From here on out, all fields should be separated with <code>§</code>, in the same string. |
|- | |- | ||
| Online Players | | Online Players | ||
Line 263: | Line 283: | ||
0000030: 30 0 | 0000030: 30 0 | ||
+ | [[Category:Protocol Details]] | ||
[[Category:Minecraft Modern]] | [[Category:Minecraft Modern]] |
Revision as of 18:00, 5 August 2024
Server List Ping (SLP) is an interface provided by Minecraft servers which supports querying the MOTD, player count, max players and server version via the usual port. SLP is part of the Protocol, but it can be disabled. The Notchian client uses this interface to display the multiplayer server list, hence the name. The SLP process changed in Java Edition 1.7 in a non-backwards compatible way, but modern clients and servers still support both the new and old process.
Contents
Current (1.7+)
This uses the regular client-server protocol. For the general packet format, see that article.
Handshake
First, the client sends a Handshake packet with its state set to 1.
Packet ID | Field Name | Field Type | Notes |
---|---|---|---|
0x00 | Protocol Version | VarInt | See protocol version numbers. The version that the client plans on using to connect to the server (which is not important for the ping). If the client is pinging to determine what version to use, by convention -1 should be set.
Setting invalid (nonexistent) version as the protocol version might cause some servers to close connection after this packet
|
Server Address | String | Hostname or IP, e.g. localhost or 127.0.0.1, that was used to connect. The Notchian server does not use this information. Note that SRV records are a complete redirect, e.g. if _minecraft._tcp.example.com points to mc.example.org, users connecting to example.com will provide mc.example.org as server address in addition to connecting to it. | |
Server Port | Unsigned Short | Default is 25565. The Notchian server does not use this information. | |
Next state | VarInt | Should be 1 for status, but could also be 2 for login. |
Status Request
The client follows up with a Status Request packet. This packet has no fields. The client is also able to skip this part entirely and send a Ping Request instead.
Packet ID | Field Name | Field Type | Notes |
---|---|---|---|
0x00 | no fields |
Status Response
The server should respond with a Status Response packet. Note that Notchian servers will for unknown reasons wait to receive the following Ping Request packet for 30 seconds before timing out and sending Response.
Packet ID | Field Name | Field Type | Notes |
---|---|---|---|
0x00 | JSON Response | String | See below; as with all strings this is prefixed by its length as a VarInt(3-byte max) |
The JSON Response field is a JSON object which has the following format:
{
"version": {
"name": "1.19.4",
"protocol": 762
},
"players": {
"max": 100,
"online": 5,
"sample": [
{
"name": "thinkofdeath",
"id": "4566e69f-c907-48ee-8d71-d7ba5aa00d20"
}
]
},
"description": {
"text": "Hello, world!"
},
"favicon": "data:image/png;base64,<data>",
"enforcesSecureChat": false
}
The name field of the version object should be considered mandatory. On newer Notchian client versions (1.20+?), it may be omitted and will be treated as though it were the string "Old" if so.
The description field is optional. If specified, it should be a text component. Note that the Notchian server has no way of providing actual text component data; instead section sign-based codes are embedded within the text of the object. However, third-party servers such as Spigot and Paper will return full components, so make sure you can handle both.
The favicon field is optional. If specified, it should be a PNG image that is Base64 encoded (without newlines: \n
, new lines no longer work since 1.13) and prepended with data:image/png;base64,
. It should also be noted that the source image must be exactly 64x64 pixels, otherwise the Notchian client will not render the image.
The players field is optional. If omitted, "???" will be displayed in dark grey in place of the player count.
The numbers max and online in the players field have a maximum supported value of 231-1 (2,147,483,647) by the vanilla Minecraft server. Larger values were observed to be set to the default value of 20.
If the sample field of the players object is present, the player names (but not their UUIDs) will be shown in order in the tooltip when hovering over the player count (or version name if incompatible). If empty, missing, or the wrong type, no tooltip will appear. Notchian servers will omit this field if there are no players on the server or if hide-online-players
is false. There doesn't appear to be a hard limit on how many players may be specified.
If a client has "Allow Server Listings" set to "OFF", Notchian servers will report their name as "Anonymous Player" and UUID as "00000000-0000-0000-0000-000000000000". This is not reliable; making a request before the player has fully joined may cause their choice to not be respected.
On most Notchian client versions, UUIDs are mandatory and must be well formed even though it is unused by Notchian clients. Newer versions (1.20+?) will tolerate malformed or missing UUIDs and act as though sample was omitted if so.
If the server has No Chat Reports installed and hasn't disabled this feature, an additional field preventsChatReports will be added to the end with a value of true causing clients with No Chat Reports to display an icon indicating the server is a "Safe Server".
After receiving the Response packet, the client may send the next packet to help calculate the server's latency, or if it is only interested in the above information it can disconnect here.
If the client does not receive a properly formatted response (e.g. a field is missing or the wrong type), it will close the socket and instead attempt a legacy ping.
Ping Request
If the process is continued, the client will now send a Ping Request packet containing some payload which is not important.
Packet ID | Field Name | Field Type | Notes |
---|---|---|---|
0x01 | Payload | Long | May be any number. Notchian clients use a system-dependent time value which is counted in milliseconds. |
Pong Response
The server will respond with the Pong Response packet and then close the connection.
Packet ID | Field Name | Field Type | Notes |
---|---|---|---|
0x01 | Payload | Long | Should be the same as sent by the client |
Ping via LAN (Open to LAN in Singleplayer)
In Singleplayer there is a function called "Open to LAN". Minecraft (in the serverlist) binds a UDP port and listens for connections to 224.0.2.60:4445
(Yes, that is the actual IP, no matter in what network you are or what your local IP Address is). If you click on "Open to LAN" Minecraft sends a packet every 1.5 seconds to the address: [MOTD]{motd}[/MOTD][AD]{port}[/AD]
. Minecraft seems to check for the following Strings: [MOTD]
, [/MOTD]
, [AD]
, [/AD]
. Anything you write outside of each of the tags will be ignored. Color codes may be used in the MOTD. Between [AD]
and [/AD]
is the servers port. If it is not numeric, 25565 will be used. If it is out of range, an error is being displayed when trying to connect. The IP Address is the same as the senders one. The string is encoded with UTF-8.
To implement it server side, just send a packet with the text (payload) to 224.0.2.60:4445
. If you are client side, bind a UDP socket and listen for connections. You can use a MulticastSocket
for that.
Examples
- C#, C#
- Java
- Python
- Rust (Server)
- MCClient-lib (Python3)
- Python3
- PHP
- LAN Server Listener (Java)
- LAN Server Listener (Kotlin)
- Swift
- Go
- Go
- Node.js
- C
- Go
1.6
This uses a protocol which is compatible with the client-server protocol as it was before the Netty rewrite. Modern servers recognize this protocol by the starting byte of fe
instead of the usual 00
.
Client to server
The client initiates a TCP connection to the server on the standard port. Instead of doing auth and logging in (as detailed in Protocol and Protocol Encryption), it sends the following data, expressed in hexadecimal:
FE
— packet identifier for a server list ping01
— server list ping's payload (always 1)FA
— packet identifier for a plugin message00 0B
— length of following string, in characters, as a short (always 11)00 4D 00 43 00 7C 00 50 00 69 00 6E 00 67 00 48 00 6F 00 73 00 74
— the stringMC|PingHost
encoded as a UTF-16BE stringXX XX
— length of the rest of the data, as a short. Compute as7 + len(hostname)
, wherelen(hostname)
is the number of bytes in the UTF-16BE encoded hostname.XX
— protocol version, e.g.4a
for the last version (74)XX XX
— length of following string, in characters, as a short...
— hostname the client is connecting to, encoded as a UTF-16BE stringXX XX XX XX
— port the client is connecting to, as an int.
All data types are big-endian.
Example packet dump:
0000000: fe01 fa00 0b00 4d00 4300 7c00 5000 6900 ......M.C.|.P.i. 0000010: 6e00 6700 4800 6f00 7300 7400 1949 0009 n.g.H.o.s.t..I.. 0000020: 006c 006f 0063 0061 006c 0068 006f 0073 .l.o.c.a.l.h.o.s 0000030: 0074 0000 63dd .t..c.
Note: All notchian servers only cares about the first 3 bytes. After reading FE 01 FA
, the response will be sent to the client. For backward compatibility, you could only send these 3 bytes and all legacy servers(<=1.6) will respond correspondingly.
Server to client
The server responds with a 0xFF kick packet. The packet begins with a single byte identifier ff
, then a two-byte big endian short giving the length of the following string in characters. You can actually ignore the length because the server closes the connection after the response is sent.
After the first 3 bytes, the packet is a UTF-16BE string. It begins with two characters: §1
, followed by a null character. On the wire these look like 00 a7 00 31 00 00
.
The remainder is null character (that is 00 00
) delimited fields:
- Protocol version (e.g.
47
) - Minecraft server version (e.g.
1.4.2
) - Message of the day (e.g.
A Minecraft Server
) - Current player count
- Max players
The entire packet looks something like this:
<---> first character 0000000: ff00 2300 a700 3100 0000 3400 3700 0000 ....§.1...4.7... 0000010: 3100 2e00 3400 2e00 3200 0000 4100 2000 1...4...2...A. . 0000020: 4d00 6900 6e00 6500 6300 7200 6100 6600 M.i.n.e.c.r.a.f. 0000030: 7400 2000 5300 6500 7200 7600 6500 7200 t. .S.e.r.v.e.r. 0000040: 0000 3000 0000 3200 30 ..0...2.0
Note: When using this protocol with servers on version 1.7.x and above, the protocol version (first field) in the response will always be 127
which is not a real protocol number, so older clients will always consider this server incompatible.
Examples
1.4 to 1.5
Prior to the Minecraft 1.6, the client to server operation is much simpler, and only sends FE 01
, with none of the following data.
Examples
Beta 1.8 to 1.3
Prior to Minecraft 1.4, the client only sends FE
.
The server should respond with a kick packet:
Field Name | Field Type | Notes |
---|---|---|
Packet Id | Byte | The kick packet id: 0xFF
|
Packet Length | Short | The length of the following string in characters (NOT BYTES). Max is 256. |
MOTD | A UTF-16BE encoded string | From here on out, all fields should be separated with § , in the same string.
|
Online Players | The count of players currently on the server. | |
Max Players | The maximum amount of players that the server is willing to accept. |
The entire packet looks something like this on wire:
<---> first character 0000000: ff00 1700 4100 2000 4d00 6900 6e00 6500 ....A. .M.i.n.e. 0000010: 6300 7200 6100 6600 7400 2000 5300 6500 c.r.a.f.t. .S.e. 0000020: 7200 7600 6500 7200 a700 3000 a700 3100 r.v.e.r.§.0.§.1. 0000030: 30 0