2013-01-04 01:45:57 -05:00
|
|
|
var net = require('net')
|
|
|
|
, EventEmitter = require('events').EventEmitter
|
|
|
|
, util = require('util')
|
|
|
|
, protocol = require('./protocol')
|
2013-04-13 16:23:31 -04:00
|
|
|
, dns = require('dns')
|
2013-01-04 01:45:57 -05:00
|
|
|
, createPacketBuffer = protocol.createPacketBuffer
|
|
|
|
, parsePacket = protocol.parsePacket
|
2013-04-05 23:02:36 -04:00
|
|
|
, debug = protocol.debug
|
2013-11-19 18:06:57 -05:00
|
|
|
;
|
2013-01-04 01:45:57 -05:00
|
|
|
|
|
|
|
module.exports = Client;
|
|
|
|
|
2013-01-07 23:36:14 -05:00
|
|
|
function Client(isServer) {
|
2013-01-04 01:45:57 -05:00
|
|
|
EventEmitter.call(this);
|
|
|
|
|
2013-01-07 23:36:14 -05:00
|
|
|
this.isServer = !!isServer;
|
2013-01-04 01:45:57 -05:00
|
|
|
this.socket = null;
|
|
|
|
this.encryptionEnabled = false;
|
|
|
|
this.cipher = null;
|
|
|
|
this.decipher = null;
|
|
|
|
}
|
|
|
|
util.inherits(Client, EventEmitter);
|
|
|
|
|
|
|
|
Client.prototype.setSocket = function(socket) {
|
|
|
|
var self = this;
|
|
|
|
self.socket = socket;
|
|
|
|
var incomingBuffer = new Buffer(0);
|
|
|
|
self.socket.on('data', function(data) {
|
|
|
|
if (self.encryptionEnabled) data = new Buffer(self.decipher.update(data), 'binary');
|
|
|
|
incomingBuffer = Buffer.concat([incomingBuffer, data]);
|
|
|
|
var parsed, packet;
|
|
|
|
while (true) {
|
|
|
|
parsed = parsePacket(incomingBuffer, self.isServer);
|
|
|
|
if (! parsed) break;
|
2013-01-29 12:39:33 -05:00
|
|
|
if (parsed.error) {
|
|
|
|
this.emit('error', parsed.error);
|
|
|
|
this.end("ProtocolError");
|
|
|
|
return;
|
|
|
|
}
|
2013-01-04 01:45:57 -05:00
|
|
|
packet = parsed.results;
|
|
|
|
incomingBuffer = incomingBuffer.slice(parsed.size);
|
|
|
|
self.emit(packet.id, packet);
|
2013-01-04 21:57:17 -05:00
|
|
|
self.emit('packet', packet);
|
2013-01-04 01:45:57 -05:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2013-01-04 21:33:19 -05:00
|
|
|
self.socket.on('connect', function() {
|
|
|
|
self.emit('connect');
|
|
|
|
});
|
2013-02-03 15:36:55 -05:00
|
|
|
|
|
|
|
self.socket.on('error', onError);
|
|
|
|
self.socket.on('close', endSocket);
|
|
|
|
self.socket.on('end', endSocket);
|
|
|
|
self.socket.on('timeout', endSocket);
|
|
|
|
|
|
|
|
function onError(err) {
|
|
|
|
self.emit('error', err);
|
|
|
|
endSocket();
|
|
|
|
}
|
|
|
|
|
2013-02-03 22:13:15 -05:00
|
|
|
var ended = false;
|
2013-02-03 15:36:55 -05:00
|
|
|
function endSocket() {
|
2013-02-03 22:13:15 -05:00
|
|
|
if (ended) return;
|
|
|
|
ended = true;
|
2013-02-03 15:36:55 -05:00
|
|
|
self.socket.removeListener('close', endSocket);
|
|
|
|
self.socket.removeListener('end', endSocket);
|
|
|
|
self.socket.removeListener('timeout', endSocket);
|
|
|
|
self.emit('end', self._endReason);
|
|
|
|
}
|
2013-01-04 01:45:57 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
Client.prototype.connect = function(port, host) {
|
|
|
|
var self = this;
|
2013-04-14 23:09:04 -04:00
|
|
|
if (port == 25565) {
|
|
|
|
dns.resolveSrv("_minecraft._tcp." + host, function(err, addresses) {
|
|
|
|
if (addresses) {
|
|
|
|
self.setSocket(net.connect(addresses[0].port, addresses[0].name));
|
|
|
|
} else {
|
|
|
|
self.setSocket(net.connect(port, host));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
self.setSocket(net.connect(port, host));
|
|
|
|
}
|
2013-01-04 01:45:57 -05:00
|
|
|
};
|
|
|
|
|
2013-01-04 20:55:53 -05:00
|
|
|
Client.prototype.end = function(reason) {
|
|
|
|
this._endReason = reason;
|
2013-01-04 01:45:57 -05:00
|
|
|
this.socket.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
Client.prototype.write = function(packetId, params) {
|
|
|
|
var buffer = createPacketBuffer(packetId, params, this.isServer);
|
2013-04-05 23:04:47 -04:00
|
|
|
debug("writing packetId " + packetId + " (0x" + packetId.toString(16) + ")");
|
2013-04-15 00:40:09 -04:00
|
|
|
debug(params);
|
2013-01-04 01:45:57 -05:00
|
|
|
var out = this.encryptionEnabled ? new Buffer(this.cipher.update(buffer), 'binary') : buffer;
|
|
|
|
this.socket.write(out);
|
|
|
|
};
|