Skip to content

Commit 1caa2c2

Browse files
evan-gooderom1504
andauthored
Fix several bugs in villager trading (#3230)
* Fix several bugs in villager trading These changes should fix issues with trades that have more than one item in the output stack, trades that cost more than one emerald, and trades with two input items, such as trades for enchanted books and tipped arrows. * Add more tests for villager trading - Add a test trade taking 1 emerald and returning 4 glass - Add a test trade taking [36 emeralds, 1 book] and returning 1 wooden sword (any unstackable should work to accurately simulate an enchanted_book trade) - Increase maximumNbTradeUses to 12 for all test trades --------- Co-authored-by: Romain Beaumont <[email protected]>
1 parent 9865ab7 commit 1caa2c2

File tree

3 files changed

+97
-50
lines changed

3 files changed

+97
-50
lines changed

lib/plugins/inventory.js

+14-3
Original file line numberDiff line numberDiff line change
@@ -407,15 +407,20 @@ function inject (bot, { hideErrors }) {
407407
}
408408

409409
function tradeMatch (limitItem, targetItem) {
410-
return targetItem.type === limitItem.type && targetItem.count >= limitItem.count
410+
return (
411+
targetItem !== null &&
412+
limitItem !== null &&
413+
targetItem.type === limitItem.type &&
414+
targetItem.count >= limitItem.count
415+
)
411416
}
412417

413418
function expectTradeUpdate (window) {
414419
const trade = window.selectedTrade
415420
const hasItem = !!window.slots[2]
416421

417422
if (hasItem !== tradeMatch(trade.inputItem1, window.slots[0])) {
418-
if (trade.hasItems2) {
423+
if (trade.hasItem2) {
419424
return hasItem !== tradeMatch(trade.inputItem2, window.slots[1])
420425
}
421426
return true
@@ -434,7 +439,7 @@ function inject (bot, { hideErrors }) {
434439
}
435440
} else if (window.type === 'minecraft:merchant') {
436441
const toUpdate = []
437-
if (slot <= 2 && !window.selectedTrade.tradeDisabled && expectTradeUpdate(window)) {
442+
if (slot <= 1 && !window.selectedTrade.tradeDisabled && expectTradeUpdate(window)) {
438443
toUpdate.push(once(bot.currentWindow, 'updateSlot:2'))
439444
}
440445
if (slot === 2) {
@@ -443,6 +448,12 @@ function inject (bot, { hideErrors }) {
443448
}
444449
}
445450
await Promise.all(toUpdate)
451+
452+
if (slot === 2 && !window.selectedTrade.tradeDisabled && expectTradeUpdate(window)) {
453+
// After the trade goes through, if the inputs are still satisfied,
454+
// expect another update in slot 2
455+
await once(bot.currentWindow, 'updateSlot:2')
456+
}
446457
}
447458
}
448459

lib/plugins/villager.js

+22-37
Original file line numberDiff line numberDiff line change
@@ -141,8 +141,11 @@ function inject (bot, { version }) {
141141
itemCount2 = villager.count(Trade.inputItem2.type, Trade.inputItem2.metadata)
142142
hasEnoughItem2 = itemCount2 >= Trade.inputItem2.count * count
143143
}
144-
if (!hasEnoughItem1 || !hasEnoughItem2) {
145-
throw new Error('Not enough items to trade')
144+
if (!hasEnoughItem1) {
145+
throw new Error('Not enough item 1 to trade')
146+
}
147+
if (!hasEnoughItem2) {
148+
throw new Error('Not enough item 2 to trade')
146149
}
147150

148151
selectTrade(choice)
@@ -152,31 +155,12 @@ function inject (bot, { version }) {
152155
if (Trade.hasItem2) proms.push(once(villager, 'updateSlot:1'))
153156
if (bot.supportFeature('setSlotAsTransaction')) {
154157
proms.push(once(villager, 'updateSlot:2'))
155-
await new Promise((resolve, reject) => {
156-
let countOfItemOneLeftToTake = itemCount1 > 64 ? 64 : itemCount1
157-
let countOfItemTwoLeftToTake = 0
158-
if (Trade.hasItem2) {
159-
countOfItemTwoLeftToTake = itemCount2 > 64 ? 64 : itemCount2
160-
}
161-
const listener = (slot, oldItem, newItem) => {
162-
if (!(slot >= villager.inventoryStart && slot <= villager.inventoryEnd)) return
163-
if (newItem === null) {
164-
if (oldItem.type === Trade.inputItem1.type) countOfItemOneLeftToTake -= oldItem.count
165-
else if (Trade.hasItem2 && oldItem.type === Trade.inputItem2.type) countOfItemTwoLeftToTake -= oldItem.count
166-
}
167-
if (countOfItemOneLeftToTake === 0 && countOfItemTwoLeftToTake === 0) {
168-
villager.off('updateSlot', listener)
169-
resolve()
170-
}
171-
}
172-
villager.on('updateSlot', listener)
173-
})
174158
}
175159
await Promise.all(proms)
176160
}
177161

178162
for (let i = 0; i < count; i++) {
179-
await putRequirements(villager, Trade, count)
163+
await putRequirements(villager, Trade)
180164
// ToDo: See if this does anything kappa
181165
Trade.nbTradeUses++
182166
if (Trade.maximumNbTradeUses - Trade.nbTradeUses === 0) {
@@ -215,24 +199,25 @@ function inject (bot, { version }) {
215199
}
216200
}
217201

218-
async function putRequirements (window, Trade, count) {
202+
async function putRequirements (window, Trade) {
219203
const [slot1, slot2] = window.slots
220-
const { stackSize: stackSize1, type: type1, metadata: metadata1 } = Trade.inputItem1
221-
const tradeCount1 = Trade.realPrice
222-
const neededCount1 = Math.min(stackSize1, tradeCount1 * count)
223-
224-
const input1 = !slot1
225-
? neededCount1
226-
: (slot1.count < tradeCount1 ? neededCount1 - slot1.count : 0)
227-
await deposit(window, type1, metadata1, input1, 0)
204+
const { type: type1, metadata: metadata1 } = Trade.inputItem1
205+
206+
const input1 = slot1
207+
? Math.max(0, Trade.realPrice - slot1.count)
208+
: Trade.realPrice
209+
if (input1) {
210+
await deposit(window, type1, metadata1, input1, 0)
211+
}
228212
if (Trade.hasItem2) {
229-
const { count: tradeCount2, stackSize: stackSize2, type: type2, metadata: metadata2 } = Trade.inputItem2
230-
const needCount2 = Math.min(stackSize2, tradeCount2 * count)
213+
const { count: tradeCount2, type: type2, metadata: metadata2 } = Trade.inputItem2
231214

232-
const input2 = !slot2
233-
? needCount2
234-
: (slot2.count < tradeCount2 ? needCount2 - slot2.count : 0)
235-
await deposit(window, type2, metadata2, input2, 1)
215+
const input2 = slot2
216+
? Math.max(0, tradeCount2 - slot2.count)
217+
: tradeCount2
218+
if (input2) {
219+
await deposit(window, type2, metadata2, input2, 1)
220+
}
236221
}
237222
}
238223

test/externalTests/trade.js

+61-10
Original file line numberDiff line numberDiff line change
@@ -6,14 +6,20 @@ module.exports = () => async (bot) => {
66

77
const villagerType = bot.registry.entitiesByName.villager ? 'villager' : 'Villager'
88
const testFluctuations = bot.supportFeature('selectingTradeMovesItems')
9+
910
const summonCommand = bot.supportFeature('indexesVillagerRecipes')
10-
? `/summon ${villagerType} ~ ~1 ~ {NoAI:1, Offers:{Recipes:[0:{maxUses:7,buy:{id:"minecraft:emerald",Count:2},sell:{id:"minecraft:pumpkin_pie",Count:2},uses: 1},1:{maxUses:7,buy:{id:"minecraft:emerald",Count:2},buyB:{id:"minecraft:pumpkin_pie",Count:2},sell:{id:"minecraft:wheat",Count:2}, uses:1}]}}`
11-
: `/summon ${villagerType} ~ ~1 ~ {NoAI:1, Offers:{Recipes:[{maxUses:7,buy:{id:"minecraft:emerald",Count:2},sell:{id:"minecraft:pumpkin_pie",Count:2},${testFluctuations ? 'demand:60,priceMultiplier:0.05f,specialPrice:-4,' : ''}uses: 1},{maxUses:7,buy:{id:"minecraft:emerald",Count:2},buyB:{id:"minecraft:pumpkin_pie",Count:2},sell:{id:"minecraft:wheat",Count:2}, uses:1}]}}`
11+
? `/summon ${villagerType} ~ ~1 ~ {NoAI:1, Offers:{Recipes:[0:{maxUses:12,buy:{id:"minecraft:emerald",Count:2},sell:{id:"minecraft:pumpkin_pie",Count:2},uses: 1},1:{maxUses:12,buy:{id:"minecraft:emerald",Count:2},buyB:{id:"minecraft:pumpkin_pie",Count:2},sell:{id:"minecraft:wheat",Count:2}, uses:1},2:{maxUses:12,buy:{id:"minecraft:emerald",Count:1},sell:{id:"minecraft:glass",Count:4},uses: 1},3:{maxUses:12,buy:{id:"minecraft:emerald",Count:36},buyB:{id:"minecraft:book",Count:1},sell:{id:"minecraft:wooden_sword",Count:1},uses: 1}]}}`
12+
: `/summon ${villagerType} ~ ~1 ~ {NoAI:1, Offers:{Recipes:[{maxUses:12,buy:{id:"minecraft:emerald",Count:2},sell:{id:"minecraft:pumpkin_pie",Count:2},${testFluctuations ? 'demand:60,priceMultiplier:0.05f,specialPrice:-4,' : ''}uses: 1},{maxUses:12,buy:{id:"minecraft:emerald",Count:2},buyB:{id:"minecraft:pumpkin_pie",Count:2},sell:{id:"minecraft:wheat",Count:2}, uses:1},{maxUses:12,buy:{id:"minecraft:emerald",Count:1},sell:{id:"minecraft:glass",Count:4},uses: 1},{maxUses:12,buy:{id:"minecraft:emerald",Count:36},buyB:{id:"minecraft:book",Count:1},sell:{id:"minecraft:wooden_sword",Count:1},uses: 1}]}}`
1213

1314
const commandBlockPos = bot.entity.position.offset(0.5, 0, 0.5)
1415
const redstoneBlockPos = commandBlockPos.offset(1, 0, 0)
1516

16-
await bot.test.setInventorySlot(36, new Item(bot.registry.itemsByName.emerald.id, 64, 0))
17+
let shouldHaveEmeralds = 0
18+
for (let slot = 9; slot <= 17; slot += 1) {
19+
await bot.test.setInventorySlot(slot, new Item(bot.registry.itemsByName.emerald.id, 64, 0))
20+
shouldHaveEmeralds += 64
21+
}
22+
await bot.test.setInventorySlot(18, new Item(bot.registry.itemsByName.book.id, 11, 0))
1723

1824
// A command block is needed to spawn the villager due to the chat's character limit in some versions
1925
bot.test.sayEverywhere(`/setblock ${commandBlockPos.toArray().join(' ')} command_block`)
@@ -40,9 +46,10 @@ module.exports = () => async (bot) => {
4046
assert.strictEqual(output.name, 'pumpkin_pie')
4147
assert.strictEqual(output.count, 2)
4248

43-
await bot.trade(villager, 0, 6)
44-
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), testFluctuations ? 64 - 24 : 64 - 12)
45-
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.pumpkin_pie.id), 12)
49+
await bot.trade(villager, 0, 11)
50+
shouldHaveEmeralds -= testFluctuations ? (2 * 2 * 11) : (2 * 11)
51+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), shouldHaveEmeralds)
52+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.pumpkin_pie.id), 22)
4653
}
4754

4855
// Handle trade #2 -- takes [2x emerald, 2x pumpkin_pie] and returns 2x wheat
@@ -61,15 +68,59 @@ module.exports = () => async (bot) => {
6168
assert.strictEqual(output.name, 'wheat')
6269
assert.strictEqual(output.count, 2)
6370

64-
await bot.trade(villager, 1, 6)
65-
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), testFluctuations ? 64 - 36 : 64 - 24)
71+
await bot.trade(villager, 1, 11)
72+
shouldHaveEmeralds -= 11 * 2
73+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), shouldHaveEmeralds)
6674
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.pumpkin_pie.id), 0)
67-
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.wheat.id), 12)
75+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.wheat.id), 22)
76+
}
77+
78+
// Handle trade #3 -- takes 1x emerald and returns 4x glass
79+
{
80+
const trade = villager.trades[2]
81+
assert.strictEqual(trade.inputs.length, 1, 'Expected single input from villager on first trade')
82+
verifyTrade(trade)
83+
84+
const [input] = trade.inputs
85+
assert.strictEqual(input.name, 'emerald')
86+
assert.strictEqual(input.count, 1)
87+
88+
const [output] = trade.outputs
89+
assert.strictEqual(output.name, 'glass')
90+
assert.strictEqual(output.count, 4)
91+
92+
await bot.trade(villager, 2, 11)
93+
shouldHaveEmeralds -= 11
94+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), shouldHaveEmeralds)
95+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.glass.id), 44)
96+
}
97+
98+
// Handle trade #4 -- takes [36x emerald, 1x book] and returns 1x wooden sword
99+
{
100+
const trade = villager.trades[3]
101+
assert.strictEqual(trade.inputs.length, 2, 'Expected two inputs from villager on second trade')
102+
verifyTrade(trade)
103+
104+
const [input1, input2] = trade.inputs
105+
assert.strictEqual(input1.name, 'emerald')
106+
assert.strictEqual(input1.count, 36)
107+
assert.strictEqual(input2.name, 'book')
108+
assert.strictEqual(input2.count, 1)
109+
110+
const [output] = trade.outputs
111+
assert.strictEqual(output.name, 'wooden_sword')
112+
assert.strictEqual(output.count, 1)
113+
114+
await bot.trade(villager, 3, 11)
115+
shouldHaveEmeralds -= 11 * 36
116+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.emerald.id), shouldHaveEmeralds)
117+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.book.id), 0)
118+
assert.strictEqual(bot.currentWindow.count(bot.registry.itemsByName.wooden_sword.id), 11)
68119
}
69120

70121
function verifyTrade (trade) {
71122
assert.strictEqual(trade.nbTradeUses, 1)
72-
assert.strictEqual(trade.maximumNbTradeUses, 7)
123+
assert.strictEqual(trade.maximumNbTradeUses, 12)
73124
assert.strictEqual(trade.tradeDisabled, false)
74125

75126
const printCountInv = function (item) {

0 commit comments

Comments
 (0)