Skip to content

Commit 2ff9919

Browse files
rom1504crux153wgaylordextremeheat
authored
Minecraft 1.20.2 support (#3262)
* Implement configuration.registry_data packet handling * Fix player entities in >= 1.20.2 * add 1.20.2 in tested versions * Add Chunk Batch support (#3263) * Add Chunk Batch support * Fix linting errors. * I really hate the linter right now. * Comment what chunkbatches are for. * Update chunk batch calculation to use Vanilla's calc but updated for Milliseconds vs Nanoseconds * Fix usage of variables starting with capital letters * Update package.json dump mcdata * server on `login` -> `playerJoin` * try CI on node20 * node20 on mc test * revert CI to node18 * update entity handling * fix typo * update internal tests * update internalTest to use feature * Update README.md * update pathfinder dep in example and add npmrc in examples --------- Co-authored-by: Crux <[email protected]> Co-authored-by: William Gaylord <[email protected]> Co-authored-by: extremeheat <[email protected]>
1 parent 5c71edf commit 2ff9919

File tree

13 files changed

+170
-86
lines changed

13 files changed

+170
-86
lines changed

docs/README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ First time using Node.js? You may want to start with the [tutorial](tutorial.md)
1717

1818
## Features
1919

20-
* Supports Minecraft 1.8 to 1.20.1 (1.8, 1.9, 1.10, 1.11, 1.12, 1.13, 1.14, 1.15, 1.16, 1.17, 1.18, 1.19 and 1.20)
20+
* Supports Minecraft 1.8 to 1.20.2 (1.8, 1.9, 1.10, 1.11, 1.12, 1.13, 1.14, 1.15, 1.16, 1.17, 1.18, 1.19 and 1.20)
2121
* Entity knowledge and tracking.
2222
* Block knowledge. You can query the world around you. Milliseconds to find any block.
2323
* Physics and movement - handle all bounding boxes

examples/anvil_saver/.npmrc

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
engine-strict=true
2+
package-lock=false

examples/pathfinder/.npmrc

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
engine-strict=true
2+
package-lock=false

examples/pathfinder/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
"version": "0.0.0",
44
"private": true,
55
"dependencies": {
6-
"mineflayer-pathfinder": "^1.6.1",
6+
"mineflayer-pathfinder": "^2.4.5",
77
"mineflayer": "file:../../"
88
},
99
"description": "A mineflayer example"

examples/place_end_crystal/.npmrc

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
engine-strict=true
2+
package-lock=false
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
engine-strict=true
2+
package-lock=false

examples/viewer/.npmrc

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
engine-strict=true
2+
package-lock=false

lib/plugins/blocks.js

+24
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,30 @@ function inject (bot, { version, storageBuilder, hideErrors }) {
268268
}
269269
})
270270

271+
// Chunk batches are used by the server to throttle the chunks per tick for players based on their connection speed.
272+
let chunkBatchStartTime = 0
273+
// The Vanilla client uses nano seconds with its weighted average starting at 2000000 converted to milliseconds that is 2
274+
let weightedAverage = 2
275+
// This is used for keeping track of the weight of the old average when updating it.
276+
let oldSampleWeight = 1
277+
278+
bot._client.on('chunk_batch_start', (packet) => {
279+
// Get the time the chunk batch is starting.
280+
chunkBatchStartTime = Date.now()
281+
})
282+
283+
bot._client.on('chunk_batch_finished', (packet) => {
284+
const milliPerChunk = (Date.now() - chunkBatchStartTime) / packet.batchSize
285+
// Prevents the MilliPerChunk from being hugely different then the average, Vanilla uses 3 as a constant here.
286+
const clampedMilliPerChunk = Math.min(Math.max(milliPerChunk, weightedAverage / 3.0), weightedAverage * 3.0)
287+
weightedAverage = ((weightedAverage * oldSampleWeight) + clampedMilliPerChunk) / (oldSampleWeight + 1)
288+
// 49 is used in Vanilla client to limit it to 50 samples
289+
oldSampleWeight = Math.min(49, oldSampleWeight + 1)
290+
bot._client.write('chunk_batch_received', {
291+
// Vanilla uses 7000000 as a constant here, since we are using milliseconds that is now 7. Not sure why they pick this constant to convert from nano seconds per chunk to chunks per tick.
292+
chunksPerTick: 7 / weightedAverage
293+
})
294+
})
271295
bot._client.on('map_chunk', (packet) => {
272296
addColumn({
273297
x: packet.x,

lib/plugins/entities.js

+52-45
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ const entityStatusEvents = {
2626
}
2727

2828
function inject (bot) {
29-
const { mobs, entitiesArray } = bot.registry
29+
const { mobs } = bot.registry
3030
const Entity = require('prismarine-entity')(bot.version)
3131
const Item = require('prismarine-item')(bot.version)
3232
const ChatMessage = require('prismarine-chat')(bot.registry)
@@ -129,33 +129,6 @@ function inject (bot) {
129129
if (eventName) bot.emit(eventName, entity)
130130
})
131131

132-
bot._client.on('named_entity_spawn', (packet) => {
133-
// in case player_info packet was not sent before named_entity_spawn : ignore named_entity_spawn (see #213)
134-
if (packet.playerUUID in bot.uuidToUsername) {
135-
// spawn named entity
136-
const entity = fetchEntity(packet.entityId)
137-
entity.type = 'player'
138-
entity.name = 'player'
139-
entity.username = bot.uuidToUsername[packet.playerUUID]
140-
entity.uuid = packet.playerUUID
141-
entity.dataBlobs = packet.data
142-
if (bot.supportFeature('fixedPointPosition')) {
143-
entity.position.set(packet.x / 32, packet.y / 32, packet.z / 32)
144-
} else if (bot.supportFeature('doublePosition')) {
145-
entity.position.set(packet.x, packet.y, packet.z)
146-
}
147-
entity.yaw = conv.fromNotchianYawByte(packet.yaw)
148-
entity.pitch = conv.fromNotchianPitchByte(packet.pitch)
149-
entity.height = NAMED_ENTITY_HEIGHT
150-
entity.width = NAMED_ENTITY_WIDTH
151-
entity.metadata = parseMetadata(packet.metadata, entity.metadata)
152-
if (bot.players[entity.username] !== undefined && !bot.players[entity.username].entity) {
153-
bot.players[entity.username].entity = entity
154-
}
155-
bot.emit('entitySpawn', entity)
156-
}
157-
})
158-
159132
bot.on('entityCrouch', (entity) => {
160133
entity.height = CROUCH_HEIGHT
161134
})
@@ -171,10 +144,11 @@ function inject (bot) {
171144
bot.emit('playerCollect', collector, collected)
172145
})
173146

147+
// What is internalId?
148+
const entityDataByInternalId = Object.fromEntries(bot.registry.entitiesArray.map((e) => [e.internalId, e]))
149+
174150
function setEntityData (entity, type, entityData) {
175-
if (entityData === undefined) {
176-
entityData = entitiesArray.find(entity => entity.internalId === type)
177-
}
151+
entityData ??= entityDataByInternalId[type]
178152
if (entityData) {
179153
entity.type = entityData.type || 'object'
180154
entity.displayName = entityData.displayName
@@ -193,24 +167,57 @@ function inject (bot) {
193167
}
194168
}
195169

196-
// spawn object/vehicle on versions < 1.19, on versions > 1.19 handles all non-player entities
197-
bot._client.on('spawn_entity', (packet) => {
198-
const entity = fetchEntity(packet.entityId)
199-
const entityData = bot.registry.entities[packet.type]
200-
setEntityData(entity, packet.type, entityData)
201-
170+
function updateEntityPos (entity, pos) {
202171
if (bot.supportFeature('fixedPointPosition')) {
203-
entity.position.set(packet.x / 32, packet.y / 32, packet.z / 32)
172+
entity.position.set(pos.x / 32, pos.y / 32, pos.z / 32)
204173
} else if (bot.supportFeature('doublePosition')) {
205-
entity.position.set(packet.x, packet.y, packet.z)
206-
} else if (bot.supportFeature('consolidatedEntitySpawnPacket')) {
207-
entity.headPitch = conv.fromNotchianPitchByte(packet.headPitch)
174+
entity.position.set(pos.x, pos.y, pos.z)
208175
}
176+
entity.yaw = conv.fromNotchianYawByte(pos.yaw)
177+
entity.pitch = conv.fromNotchianPitchByte(pos.pitch)
178+
}
209179

210-
entity.uuid = packet.objectUUID
211-
entity.yaw = conv.fromNotchianYawByte(packet.yaw)
212-
entity.pitch = conv.fromNotchianPitchByte(packet.pitch)
213-
entity.objectData = packet.objectData
180+
function addNewPlayer (entityId, uuid, pos) {
181+
const entity = fetchEntity(entityId)
182+
entity.type = 'player'
183+
entity.name = 'player'
184+
entity.username = bot.uuidToUsername[uuid]
185+
entity.uuid = uuid
186+
updateEntityPos(entity, pos)
187+
entity.height = NAMED_ENTITY_HEIGHT
188+
entity.width = NAMED_ENTITY_WIDTH
189+
if (bot.players[entity.username] !== undefined && !bot.players[entity.username].entity) {
190+
bot.players[entity.username].entity = entity
191+
}
192+
return entity
193+
}
194+
195+
function addNewNonPlayer (entityId, uuid, entityType, pos) {
196+
const entity = fetchEntity(entityId)
197+
const entityData = bot.registry.entities[entityType]
198+
setEntityData(entity, entityType, entityData)
199+
updateEntityPos(entity, pos)
200+
return entity
201+
}
202+
203+
bot._client.on('named_entity_spawn', (packet) => {
204+
// in case player_info packet was not sent before named_entity_spawn : ignore named_entity_spawn (see #213)
205+
if (packet.playerUUID in bot.uuidToUsername) {
206+
// spawn named entity
207+
const entity = addNewPlayer(packet.entityId, packet.playerUUID, packet, packet.metadata)
208+
entity.dataBlobs = packet.data // this field doesn't appear to be listed on any version
209+
entity.metadata = parseMetadata(packet.metadata, entity.metadata) // 1.8
210+
bot.emit('entitySpawn', entity)
211+
}
212+
})
213+
214+
// spawn object/vehicle on versions < 1.19, on versions > 1.19 handles all non-player entities
215+
// on versions >= 1.20.2, this also handles player entities
216+
bot._client.on('spawn_entity', (packet) => {
217+
const entityData = entityDataByInternalId[packet.type]
218+
const entity = entityData?.type === 'player'
219+
? addNewPlayer(packet.entityId, packet.objectUUID, packet)
220+
: addNewNonPlayer(packet.entityId, packet.objectUUID, packet.type, packet)
214221
bot.emit('entitySpawn', entity)
215222
})
216223

lib/plugins/game.js

+5
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,11 @@ function inject (bot, options) {
7171
const brandChannel = getBrandCustomChannelName()
7272
bot._client.registerChannel(brandChannel, ['string', []])
7373

74+
// 1.20.2
75+
bot._client.on('registry_data', (packet) => {
76+
bot.registry.loadDimensionCodec(packet.codec)
77+
})
78+
7479
bot._client.on('login', (packet) => {
7580
handleRespawnPacketData(packet)
7681

lib/version.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
1-
const testedVersions = ['1.8.8', '1.9.4', '1.10.2', '1.11.2', '1.12.2', '1.13.2', '1.14.4', '1.15.2', '1.16.5', '1.17.1', '1.18.2', '1.19', '1.19.2', '1.19.3', '1.19.4', '1.20.1']
1+
const testedVersions = ['1.8.8', '1.9.4', '1.10.2', '1.11.2', '1.12.2', '1.13.2', '1.14.4', '1.15.2', '1.16.5', '1.17.1', '1.18.2', '1.19', '1.19.2', '1.19.3', '1.19.4', '1.20.1', '1.20.2']
22
module.exports = {
3+
34
testedVersions,
45
latestSupportedVersion: testedVersions[testedVersions.length - 1],
56
oldestSupportedVersion: testedVersions[0]
7+
68
}

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
},
2222
"license": "MIT",
2323
"dependencies": {
24-
"minecraft-data": "^3.44.0",
24+
"minecraft-data": "^3.56.0",
2525
"minecraft-protocol": "^1.44.0",
2626
"prismarine-biome": "^1.1.1",
2727
"prismarine-block": "^1.17.0",

0 commit comments

Comments
 (0)