const mc = require('minecraft-protocol') const states = mc.states function printHelpAndExit(exitCode) { console.log('usage: node proxy.js [...] ') console.log('options:') console.log(' --dump name') console.log(' print to stdout messages with the specified name.') console.log(' --dump-all') console.log(' print to stdout all messages, except those specified with -x.') console.log(' -x name') console.log(' do not print messages with this name.') console.log(' name') console.log(' a packet name as defined in protocol.json') process.exit(exitCode) } if (process.argv.length < 4) { console.log('Too few arguments!') printHelpAndExit(1) } process.argv.forEach(function(val) { if (val === '-h') { printHelpAndExit(0) } }) const args = process.argv.slice(2) let port let version let printAllNames = false const printNameWhitelist = {} const printNameBlacklist = {}; (function() { let i = 0 for (i = 0; i < args.length; i++) { const option = args[i] if (!/^-/.test(option)) break if (option === '--dump-all') { printAllNames = true continue } i++ const name = args[i] if (option === '--dump') { printNameWhitelist[name] = 'io' } else if (option === '-x') { printNameBlacklist[name] = 'io' } else { printHelpAndExit(1) } } if (!(i + 2 <= args.length && args.length <= i + 4)) printHelpAndExit(1) port = args[i++] version = args[i++] })() const srv = mc.createServer({ 'online-mode': false, port, keepAlive: false, version, motd: 'Customized MOTD & Favicon', favicon: '' }) let targetServer = new Map(); srv.on('connection', client => { client.on('packet', (data, meta) => { if (meta.state === states.HANDSHAKING && client.state === states.HANDSHAKING) { console.log('[HANDSHAKING] ', client.state + '.' + meta.name + ' :' + JSON.stringify(data)) switch (data.serverHost) { case 'egg.localhost.demo': targetServer.set(client.socket.remoteAddress, { host: 'cloudegg.cloud', port: 25565 }) break case 'm.localhost.demo': targetServer.set(client.socket.remoteAddress, { host: 'Mchaptim.cn', port: 25565 }) break case 'toilet.localhost.demo': targetServer.set(client.socket.remoteAddress, { host: 'toiletmc.net', port: 25565 }) break default: targetServer.set(client.socket.remoteAddress, { host: 'localhost', port: 25566 }) } } }) }) srv.on('login', function(client) { const addr = client.socket.remoteAddress console.log('Incoming connection', '(' + addr + ')') let endedClient = false let endedremoteClient = false client.on('end', function() { endedClient = true console.log('Connection closed by client', '(' + addr + ')') if (!endedremoteClient) { remoteClient.end('End') } }) client.on('error', function(err) { endedClient = true console.log('Connection error by client', '(' + addr + ')') console.log(err.stack) if (!endedremoteClient) { remoteClient.end('Error') } }) const remoteClient = mc.createClient({ host: targetServer.get(addr).host, port: targetServer.get(addr).port, username: client.username, keepAlive: false, version }) client.on('packet', function(data, meta) { if (remoteClient.state === states.PLAY && meta.state === states.PLAY) { if (shouldDump(meta.name, 'o')) { console.log('client->server:', client.state + ' ' + meta.name + ' :', JSON.stringify(data)) } if (!endedremoteClient) { remoteClient.write(meta.name, data) } } }) remoteClient.on('packet', function(data, meta) { if (meta.state === states.PLAY && client.state === states.PLAY) { if (shouldDump(meta.name, 'i')) { console.log('client<-server:', remoteClient.state + '.' + meta.name + ' :' + JSON.stringify(data)) } if (!endedClient) { client.write(meta.name, data) if (meta.name === 'set_compression') { client.compressionThreshold = data.threshold } // Set compression } } }) remoteClient.on('error', err => { console.log(err.message) }) const bufferEqual = require('buffer-equal') remoteClient.on('raw', function(buffer, meta) { if (client.state !== states.PLAY || meta.state !== states.PLAY) { return } const packetData = remoteClient.deserializer.parsePacketBuffer(buffer).data.params const packetBuff = client.serializer.createPacketBuffer({ name: meta.name, params: packetData }) if (!bufferEqual(buffer, packetBuff)) { console.log('client<-server: Error in packet ' + meta.state + '.' + meta.name) console.log('received buffer', buffer.toString('hex')) console.log('produced buffer', packetBuff.toString('hex')) console.log('received length', buffer.length) console.log('produced length', packetBuff.length) } /* if (client.state === states.PLAY && brokenPackets.indexOf(packetId.value) !=== -1) { console.log(`client<-server: raw packet); console.log(packetData); if (!endedClient) client.writeRaw(buffer); } */ }) client.on('raw', function(buffer, meta) { if (meta.state !== states.PLAY || remoteClient.state !== states.PLAY) { return } const packetData = client.deserializer.parsePacketBuffer(buffer).data.params const packetBuff = remoteClient.serializer.createPacketBuffer({ name: meta.name, params: packetData }) if (!bufferEqual(buffer, packetBuff)) { console.log('client->server: Error in packet ' + meta.state + '.' + meta.name) console.log('received buffer', buffer.toString('hex')) console.log('produced buffer', packetBuff.toString('hex')) console.log('received length', buffer.length) console.log('produced length', packetBuff.length) } }) remoteClient.on('end', function() { endedremoteClient = true console.log('Connection closed by server', '(' + addr + ')') if (!endedClient) { client.end('服务器关闭了连接') } }) remoteClient.on('error', function(err) { endedremoteClient = true console.log('Connection error by server', '(' + addr + ') ', err) console.log(err.stack) if (!endedClient) { client.end('连接发生了错误') } }) }) function shouldDump(name, direction) { if (matches(printNameBlacklist[name])) return false if (printAllNames) return true return matches(printNameWhitelist[name]) function matches(result) { return result !== undefined && result !== null && result.indexOf(direction) !== -1 } }