Difference between revisions of "Server List Ping"

From wiki.vg
Jump to navigation Jump to search
m (fix login packet link)
(31 intermediate revisions by 15 users not shown)
Line 1: Line 1:
Minecraft supports querying the MOTD, player count, max players and server version via the usual port. Unlike [[Query]], the server list ping interface is always enabled.
+
'''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]], so unlike [[Query]], the interface is always enabled. 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 current servers still support both the [[#Current|new]] and [[#1.6|old]] process.
  
== 1.7 ==
+
== Current ==
  
The server ping process changed in 1.7 in a non-backwards compatible way but the 1.7 server does support both (see below)
+
This uses the regular client-server [[protocol]]. For the general packet format, see that article.
  
The packets have a length prefix which is a [https://developers.google.com/protocol-buffers/docs/encoding#varints VarInt] ([https://gist.github.com/thinkofdeath/e975ddee04e9c87faf22 Java example]). The data contained in the next ''length'' bytes is as followed:
+
=== Handshake ===
 +
 
 +
First, the client sends a [[Protocol#Handshake|Handshake]] packet with its state set to 1.
  
 
{| class="wikitable"
 
{| class="wikitable"
! Field Name !! Field Type !! Notes
+
! Packet ID
|-
+
! Field Name
| Packet ID || VarInt ||
+
! Field Type
|-
+
! Notes
| Data || ||
+
|-
|}
+
|rowspan="4"| 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 <code>-1</code> should be set.
 +
|-
 +
| 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 [[Protocol#Status|status]], but could also be 2 for [[Protocol#Login|login]].
 +
|}
  
Where data depends on the packet.
+
=== Request ===
  
=== Ping Process ===
+
The client follows up with a [[Protocol#Request|Request]] packet. This packet has no fields.
 
 
Firstly a [[Protocol#Handshake|Handshake]] packet must be sent with its state set to 1. The layout of the handshake packet is as followed
 
  
 
{| class="wikitable"
 
{| class="wikitable"
! Packet ID !! Field Name !! Field Type !! Notes
+
! Packet ID
|-
+
! Field Name
| rowspan=4 | 0x00
+
! Field Type
| Protocol Version || VarInt || (4 as of 1.7.2)
+
! Notes
|-
+
|-
| Server Address (hostname or IP) || String || A string is a VarInt length followed ''length'' bytes which make an UTF-8 string
+
| 0x00
|-
+
|colspan="3"| ''no fields''
| Server Port || Unsigned Short|| A short has 2 byte size. It should be read in Big-endian order
+
|}
|-
 
| Next state || VarInt || 1 for status
 
|}
 
 
 
Followed by a [[Protocol#Request|Status Request]] packet. The request packet has no fields
 
  
{| class="wikitable"
+
=== Response ===
! Packet ID !! Field Name !! Field Type !! Notes
 
|-
 
| rowspan=1 | 0x00 || || ||
 
|}
 
  
The server should respond with a [[Protocol#Response|Status Response]] packet
+
The server should respond with a [[Protocol#Response|Response]] packet. Note that Notchian servers will for unknown reasons wait to receive the following [[Protocol#Ping|Ping]] packet for 30 seconds before timing out and sending Response.
  
 
{| class="wikitable"
 
{| class="wikitable"
! Packet ID !! Field Name !! Field Type !! Notes
+
! Packet ID
|-
+
! Field Name
| rowspan=1 | 0x00  
+
! Field Type
| JSON Response || String || A string is a VarInt length followed ''length'' bytes which make an UTF-8 string
+
! Notes
|}
+
|-
 +
| 0x00  
 +
| JSON Response
 +
| String
 +
| see below
 +
  |}
  
The response is a json object which has the following format
+
The JSON Response field is a [[wikipedia:JSON|JSON]] object which has the following format:
  
<pre>
+
<syntaxhighlight lang="javascript">
 
{
 
{
"version": {
+
    "version": {
"name": "1.7.2",
+
        "name": "1.8.7",
"protocol": 4
+
        "protocol": 47
},
+
    },
"players": {
+
    "players": {
"max": 100,
+
        "max": 100,
"online": 5,
+
        "online": 5,
"sample":[
+
        "sample": [
{"name":"Thinkofdeath", "id":"4566e69fc90748ee8d71d7ba5aa00d20"}
+
            {
]
+
                "name": "thinkofdeath",
},
+
                "id": "4566e69f-c907-48ee-8d71-d7ba5aa00d20"
"description": {"text":"Hello world"},
+
            }
"favicon": "data:image/png;base64,<data>"
+
        ]
 +
    },
 +
    "description": {
 +
        "text": "Hello world"
 +
    },
 +
    "favicon": "data:image/png;base64,<data>"
 
}
 
}
</pre>
+
</syntaxhighlight>
  
The ''sample'' and ''favicon'' sections are optional.
+
The ''description'' field is a [[Chat]] object.  Note that the Notchian server has no way of providing actual chat component data; instead section sign-based codes are embedded within the text of text of the object.
  
The ''favicon'' should be a png image that is [http://en.wikipedia.org/wiki/Base64 Base64] encoded and prepended with 'data:image/png;base64,'
+
The ''favicon'' field is optional. The ''sample'' field must be set, but can be empty.
  
The client sends another packet to help calculate the server's latency but if you only wanted the above information you can stop here.
+
The ''favicon'' 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>.
  
The client will then send a [[Protocol#Ping|Ping]] packet containing the time that the ping was sent.
+
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, then it will instead attempt a [[Server_List_Ping#1.6|legacy ping]].
 +
 
 +
=== Ping ===
 +
 
 +
If the process is continued, the client will now send a [[Protocol#Ping|Ping]] packet containing some payload which is not important.
  
 
{| class="wikitable"
 
{| class="wikitable"
! Packet ID !! Field Name !! Field Type !! Notes
+
! Packet ID
|-
+
! Field Name
| rowspan=1 | 0x01  
+
! Field Type
| Time || Long ||
+
! Notes
|}
+
|-
 +
| 0x01
 +
| Payload
 +
| Long
 +
| May be any number. Notchian clients use a system-dependent time value which is counted in milliseconds.
 +
|}
  
The server will respond with the same packet and then close the connection
+
=== Pong ===
 +
 
 +
The server will respond with the [[Protocol#Pong|Pong]] packet and then close the connection.
  
 
{| class="wikitable"
 
{| class="wikitable"
! Packet ID !! Field Name !! Field Type !! Notes
+
! Packet ID
|-
+
! Field Name
| rowspan=1 | 0x01  
+
! Field Type
| Time || Long ||
+
! Notes
|}
+
|-
 +
| 0x01
 +
| Payload
 +
| Long
 +
| Should be the same as sent by the client
 +
|}
  
 
=== Examples ===
 
=== Examples ===
  
 +
* [https://gist.github.com/csh/2480d14fbbb33b4bbae3 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://gist.github.com/ewized/97814f57ac85af7128bf Python3]
 +
* [https://github.com/winny-/mcstat PHP]
  
 
== 1.6 ==
 
== 1.6 ==
  
=== Client -> Server ===
+
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 <code>fe</code> instead of the usual <code>00</code>.
  
The client initiates a TCP connection to the minecraft server on the standard port. Instead of doing auth and logging in (as detailed in [[Protocol Encryption]]), it sends the following data, expressed in hexadecimal:
+
=== Client to server ===
  
# <code>FE</code> - packet identifier for a [[Protocol#0xFE|server list ping]]
+
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:
# <code>01</code> - server list ping's payload (always 1)
 
# <code>FA</code> - packet identifier for a [[Protocol#0xFA|plugin message]]
 
# <code>00 0B</code> - length of following string, in characters, as a short (always 11)
 
# <code>00 4D 00 43 00 7C 00 50 00 69 00 6E 00 67 00 48 00 6F 00 73 00 74</code> - the string "MC|PingHost" encoded as a [http://en.wikipedia.org/wiki/UTF-16 UTF-16BE] string
 
# <code>XX XX</code> - length of the rest of the data, as a short. Compute as <code>7 + 2*len(hostname)</code>
 
# <code>XX</code> - protocol version - currently 74 (decimal)
 
# <code>XX XX</code> - length of following string, in characters, as a short
 
# <code>...</code> - hostname the client is connecting to, encoded the same way as "MC|PingHost"
 
# <code>XX XX XX XX</code> - port the client is connecting to, as an int.
 
  
All [[Data_Types|data types]] are big-endian.
+
# <code>FE</code> — packet identifier for a server list ping
 +
# <code>01</code> — server list ping's payload (always 1)
 +
# <code>FA</code> — packet identifier for a plugin message
 +
# <code>00 0B</code> — length of following string, in characters, as a short (always 11)
 +
# <code>00 4D 00 43 00 7C 00 50 00 69 00 6E 00 67 00 48 00 6F 00 73 00 74</code> — the string <code>MC|PingHost</code> encoded as a [http://en.wikipedia.org/wiki/UTF-16 UTF-16BE] string
 +
# <code>XX XX</code> — length of the rest of the data, as a short. Compute as <code>7 + len(hostname)</code>, where <code>len(hostname)</code> is the number of bytes in the UTF-16BE encoded hostname.
 +
# <code>XX</code> — [[Protocol version numbers#Versions before the Netty rewrite|protocol version]], e.g. <code>4a</code> for the last version (74)
 +
# <code>XX XX</code> — length of following string, in characters, as a short
 +
# <code>...</code> — hostname the client is connecting to, encoded as a [http://en.wikipedia.org/wiki/UTF-16 UTF-16BE] string
 +
# <code>XX XX XX XX</code> — port the client is connecting to, as an int.
  
Packet dump:
+
All data types are big-endian.
  
    0000000: fe01 fa00 0b00 4d00 4300 7c00 5000 6900  ......M.C.|.P.i.
+
Example packet dump:
    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.
 
  
=== Server -> Client ===
+
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.
  
The server responds with a [[Protocol#Disconnect.2FKick_.280xFF.29|0xFF kick]] packet. The packet begins with a single byte identifier <code>ff</code>, then a two-byte big endian short giving the length of the proceeding string in characters. You can actually ignore the length because the server closes the connection after the response is sent.
+
=== Server to client ===
 +
 
 +
The server responds with a 0xFF kick packet. The packet begins with a single byte identifier <code>ff</code>, 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: <code>§1</code>, followed by a null character. On the wire these look like <code>00 a7 00 31 00 00</code>.
 
After the first 3 bytes, the packet is a UTF-16BE string. It begins with two characters: <code>§1</code>, followed by a null character. On the wire these look like <code>00 a7 00 31 00 00</code>.
Line 134: Line 174:
 
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>47</code>)
+
# Protocol version (e.g. <code>74</code>)
# Minecraft server version (e.g. <code>1.4.2</code>)
+
# Minecraft server version (e.g. <code>1.8.7</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 142: Line 182:
 
The entire packet looks something like this:
 
The entire packet looks something like this:
  
                    <---> first character
+
                <---> first character
    0000000: ff00 2300 a700 3100 0000 3400 3700 0000  ....§.1...4.7...
+
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. .
+
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.
+
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.
+
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
+
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 <code>127</code> which is not a real protocol number, so older clients will always consider this server incompatible.
  
 
=== Examples ===
 
=== Examples ===
  
 
* [https://gist.github.com/6281388 Ruby]
 
* [https://gist.github.com/6281388 Ruby]
 +
* [https://github.com/winny-/mcstat PHP]
  
== 1.4 - 1.5 ==
+
== 1.4 to 1.5 ==
  
Prior to the Minecraft 1.6, the client -> server operation is much simpler, and only sends <code>FE 01</code>, with none of the following data beginning <code>FA ...</code>
+
Prior to the Minecraft 1.6, the client to server operation is much simpler, and only sends <code>FE 01</code>, with none of the following data.
  
 
=== Examples ===
 
=== Examples ===
Line 163: Line 206:
 
* [https://gist.github.com/6223787 C#]
 
* [https://gist.github.com/6223787 C#]
  
== Beta 1.8 - 1.3 ==
+
== Beta 1.8 to 1.3 ==
  
 
Prior to Minecraft 1.4, the client only sends <code>FE</code>.  
 
Prior to Minecraft 1.4, the client only sends <code>FE</code>.  
Line 175: Line 218:
 
The entire packet looks something like this:
 
The entire packet looks something like this:
  
                    <---> first character
+
                <---> first character
    0000000: ff00 1700 4100 2000 4d00 6900 6e00 6500  ....A. .M.i.n.e.
+
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.
+
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.
+
0000020: 7200 7600 6500 7200 a700 3000 a700 3100  r.v.e.r.§.0.§.1.
    0000030: 30                                      0
+
0000030: 30                                      0
 +
 
 +
[[Category:Minecraft Modern]]

Revision as of 22:07, 13 August 2019

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, so unlike Query, the interface is always enabled. The Notchian client uses this interface to display the multiplayer server list, hence the name. The SLP process changed in 1.7 in a non-backwards compatible way, but current servers still support both the new and old process.

Current

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.
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.

Request

The client follows up with a Request packet. This packet has no fields.

Packet ID Field Name Field Type Notes
0x00 no fields

Response

The server should respond with a Response packet. Note that Notchian servers will for unknown reasons wait to receive the following Ping packet for 30 seconds before timing out and sending Response.

Packet ID Field Name Field Type Notes
0x00 JSON Response String see below

The JSON Response field is a JSON object which has the following format:

{
    "version": {
        "name": "1.8.7",
        "protocol": 47
    },
    "players": {
        "max": 100,
        "online": 5,
        "sample": [
            {
                "name": "thinkofdeath",
                "id": "4566e69f-c907-48ee-8d71-d7ba5aa00d20"
            }
        ]
    },	
    "description": {
        "text": "Hello world"
    },
    "favicon": "data:image/png;base64,<data>"
}

The description field is a Chat object. Note that the Notchian server has no way of providing actual chat component data; instead section sign-based codes are embedded within the text of text of the object.

The favicon field is optional. The sample field must be set, but can be empty.

The favicon 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,.

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, then it will instead attempt a legacy ping.

Ping

If the process is continued, the client will now send a Ping 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

The server will respond with the Pong 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

Examples

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:

  1. FE — packet identifier for a server list ping
  2. 01 — server list ping's payload (always 1)
  3. FA — packet identifier for a plugin message
  4. 00 0B — length of following string, in characters, as a short (always 11)
  5. 00 4D 00 43 00 7C 00 50 00 69 00 6E 00 67 00 48 00 6F 00 73 00 74 — the string MC|PingHost encoded as a UTF-16BE string
  6. XX XX — length of the rest of the data, as a short. Compute as 7 + len(hostname), where len(hostname) is the number of bytes in the UTF-16BE encoded hostname.
  7. XXprotocol version, e.g. 4a for the last version (74)
  8. XX XX — length of following string, in characters, as a short
  9. ... — hostname the client is connecting to, encoded as a UTF-16BE string
  10. XX 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.

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:

  1. Protocol version (e.g. 74)
  2. Minecraft server version (e.g. 1.8.7)
  3. Message of the day (e.g. A Minecraft Server)
  4. Current player count
  5. 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.

Additionally, the response from the server only contains 3 fields delimited by §:

  1. Message of the day (e.g. A Minecraft Server)
  2. Current player count
  3. Max players

The entire packet looks something like this:

                <---> 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