mirror of
https://github.com/MeowLynxSea/Proksea.git
synced 2025-07-09 10:54:40 +00:00
200 lines
17 KiB
JavaScript
200 lines
17 KiB
JavaScript
const mc = require('minecraft-protocol')
|
|
|
|
const states = mc.states
|
|
|
|
function printHelpAndExit(exitCode) {
|
|
console.log('usage: node proxy.js [<options>...] <serverPort> <verison>')
|
|
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
|
|
}
|
|
} |