CONTRIBUTING.md
Contributions are always welcome :). If you have any questions, please discuss on the Discord or in a Discussion.
Good sources for the Minecraft bedrock protocol are gophertunnel, ClouburstMC's protocol library and PocketMine.
Protocol updates need to happen in two places: in minecraft-data to update the protocol schema (the actual data structures for the packets) and here in the protocol library side. If no changes to the underlying protocol are made aside from packet structure changes (add, remove, modify packets) then the only change needed in bedrock-protocol is to update the README documentation and some constants in src/options.js
(update the CURRENT_VERSION).
Steps to update:
- Update the protocol data in minecraft-data : see the instructions here.
- Find the relevant changes to the protocol for the current version
- Update the .YML files in minecraft-data accordingly (see the Packet serialization notes at the bottom here for info on syntax)
- Then follow the steps to build the protocol .YML files into JSON
- Do a release of the minecraft-data package
- Add the version to
src/options.js
here - Run
npm run build
andnpm test
to test that everything is OK
For development purposes, you can easily alter the protocol locally without a remote minecraft-data release :
- Run
npm install
on the root of this repo after git cloning - Open
node_modules/minecraft-data/minecraft-data/data/bedrock/latest/
and update the .YML files as you need, following the schema at the bottom (make sure to update '!version' if you are changing version) - Go back to the root of this repo and run
npm run build
. - Then
npm test
; the protocol changes should be automatically applied
For example, here is a PR for the update to 1.17.30 in minecraft-data - here is an accompanying change for bedrock-protocol.
The code structure is similar to node-minecraft-protocol. For raknet, raknet-native is used for Raknet communication.
This project uses ProtoDef to serialize and deserialize Minecraft packets. See the documentation here. The ProtoDef schema is JSON can be found here for use in other languages.
In bedrock-protocol, JavaScript code is generated from the JSON through the node-protodef compiler.
For easier maintainability, the JSON is generated from a more human readable YAML format. You can read more here. Some documentation is below.
Packets should go in proto.yml and extra types should go in types.yml.
# This defines a new data structure, a ProtoDef container.
Position:
# Variable `x` in this struct has a type of `li32`, a little-endian 32-bit integer
x: li32
# `z` is a 32-bit LE *unsigned* integer
z: lu32
# `b` is a 32-bit LE floating point
y: lf32
# Fields starting with `packet_` are structs representing Minecraft packets
packet_player_position:
# Fields starting with ! are ignored by the parser. '!id' is used by the parser when generating the packet map
!id: 0x29 # This packet is ID #0x29
!bound: client # `client` or `server` bound, just for documentation purposes. This has no other effect.
# Read `on_ground` as a boolean
on_ground: bool
# Read `position` as custom data type `Position` defined above.
position: Position
# Reads a 8-bit unsigned integer, then maps it to a string
movement_reason: u8 =>
0: player_jump
1: player_autojump
2: player_sneak
3: player_sprint
4: player_fall
# A `_` as a field name declares an anonymous data structure which will be inlined. Adding a '?' at the end will start a `switch` statement
_: movement_reason ?
# if the condition matches to the string "player_jump" or "player_autojump", there is a data struct that needs to be read
if player_jump or player_autojump:
# read `original_position` as a `Position`
original_position: Position
jump_tick: li64
# if the condition matches "player_fall", read the containing field
if player_fall:
original_position: Position
default: void
# Another way to declare a switch, without an anonymous structure. `player_hunger` will be read as a 8-bit int if movement_reason == "player_sprint"
player_hunger: movement_reason ?
if player_sprint: u8
# The default statement as in a switch statement
default: void
# Square brackets notate an array. At the left is the type of the array values, at the right is the type of
# the length prefix. If no type at the left is specified, the type is defined below.
# Reads an array of `Position`, length-prefixed with a ProtoBuf-type unsigned variable length integer (VarInt)
last_positions: Position[]varint
# Reads an array, length-prefixed with a zigzag-encoded signed VarInt
# The data structure for the array is defined underneath
keys_down: []zigzag32
up: bool
down: bool
shift: bool
The above roughly translates to the following JavaScript code to read a packet:
function read_position(stream) {
const ret = {}
ret.x = stream.readSignedInt32LE()
ret.z = stream.readUnsignedInt32LE()
ret.y = stream.readFloat32LE()
return ret
}
function read_player_position(stream) {
const ret = {}
ret.on_ground = Boolean(stream.readU8())
ret.position = read_player_position(stream)
let __movement_reason = stream.readU8()
let movement_reason = { 0: 'player_jump', 1: 'player_autojump', 2: 'player_sneak', 3: 'player_sprint', 4: 'player_fall' }[__movement_reason]
switch (movement_reason) {
case 'player_jump':
case 'player_autojump':
ret.original_position = read_player_position(stream)
ret.jump_tick = stream.readInt64LE(stream)
break
case 'player_fall':
ret.original_position = read_player_position(stream)
break
default: break
}
ret.player_hunger = undefined
if (movement_reason == 'player_sprint') ret.player_hunger = stream.readU8()
ret.last_positions = []
for (let i = 0; i < stream.readUnsignedVarInt(); i++) {
ret.last_positions.push(read_player_position(stream))
}
ret.keys_down = []
for (let i = 0; i < stream.readZigZagVarInt(); i++) {
const ret1 = {}
ret1.up = Boolean(stream.readU8())
ret1.down = Boolean(stream.readU8())
ret1.shift = Boolean(stream.readU8())
ret.keys_down.push(ret1)
}
return ret
}
and the results in the following JSON for the packet:
{
"on_ground": false,
"position": { "x": 0, "y": 2, "z": 0 },
"movement_reason": "player_jump",
"original_position": { "x": 0, "y": 0, "z": 0 },
"jump_tick": 494894984,
"last_positions": [{ "x": 0, "y": 1, "z": 0 }],
"keys_down": []
}
Custom ProtoDef types can be inlined as JSON:
string: ["pstring",{"countType":"varint"}]