2017-08-09 11:05:55 +02:00
|
|
|
const EventEmitter = require('events').EventEmitter,
|
2017-08-09 12:32:09 +02:00
|
|
|
dns = require('dns'),
|
|
|
|
net = require('net'),
|
|
|
|
async = require('async'),
|
2017-08-10 13:49:42 +02:00
|
|
|
Reader = require('../lib/reader'),
|
|
|
|
HexUtil = require('../lib/HexUtil');
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 11:05:55 +02:00
|
|
|
class Core extends EventEmitter {
|
2017-08-09 12:32:09 +02:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.options = {
|
2018-01-31 11:03:13 +01:00
|
|
|
socketTimeout: 1000,
|
|
|
|
attemptTimeout: 10000,
|
2018-01-31 07:41:57 +01:00
|
|
|
maxAttempts: 1
|
2017-08-09 12:32:09 +02:00
|
|
|
};
|
|
|
|
this.attempt = 1;
|
|
|
|
this.finished = false;
|
|
|
|
this.encoding = 'utf8';
|
|
|
|
this.byteorder = 'le';
|
|
|
|
this.delimiter = '\0';
|
2017-08-09 12:41:30 +02:00
|
|
|
this.srvRecord = null;
|
2018-01-31 11:03:13 +01:00
|
|
|
this.attemptTimeoutTimer = null;
|
2017-08-09 12:32:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fatal(err,noretry) {
|
2018-01-31 07:41:57 +01:00
|
|
|
if(!noretry && this.attempt < this.options.maxAttempts) {
|
2017-08-09 12:32:09 +02:00
|
|
|
this.attempt++;
|
|
|
|
this.start();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.done({error: err.toString()});
|
|
|
|
}
|
|
|
|
|
|
|
|
initState() {
|
|
|
|
return {
|
|
|
|
name: '',
|
|
|
|
map: '',
|
|
|
|
password: false,
|
|
|
|
|
|
|
|
raw: {},
|
|
|
|
|
|
|
|
maxplayers: 0,
|
|
|
|
players: [],
|
|
|
|
bots: []
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
finalizeState(state) {}
|
|
|
|
|
|
|
|
finish(state) {
|
|
|
|
this.finalizeState(state);
|
|
|
|
this.done(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
done(state) {
|
|
|
|
if(this.finished) return;
|
|
|
|
|
|
|
|
if(this.options.notes)
|
|
|
|
state.notes = this.options.notes;
|
|
|
|
|
|
|
|
state.query = {};
|
|
|
|
if('host' in this.options) state.query.host = this.options.host;
|
|
|
|
if('address' in this.options) state.query.address = this.options.address;
|
|
|
|
if('port' in this.options) state.query.port = this.options.port;
|
|
|
|
if('port_query' in this.options) state.query.port_query = this.options.port_query;
|
|
|
|
state.query.type = this.type;
|
|
|
|
if('pretty' in this) state.query.pretty = this.pretty;
|
2018-01-31 07:47:09 +01:00
|
|
|
state.query.duration = Date.now() - this.startMillis;
|
2018-01-31 11:03:13 +01:00
|
|
|
state.query.attempts = this.attempt;
|
2018-01-31 07:34:11 +01:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
this.reset();
|
|
|
|
this.finished = true;
|
|
|
|
this.emit('finished',state);
|
|
|
|
if(this.options.callback) this.options.callback(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
reset() {
|
2018-01-31 11:03:13 +01:00
|
|
|
clearTimeout(this.attemptTimeoutTimer);
|
2017-08-09 12:32:09 +02:00
|
|
|
if(this.timers) {
|
|
|
|
for (const timer of this.timers) {
|
|
|
|
clearTimeout(timer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.timers = [];
|
|
|
|
|
|
|
|
if(this.tcpSocket) {
|
|
|
|
this.tcpSocket.destroy();
|
|
|
|
delete this.tcpSocket;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.udpTimeoutTimer = false;
|
|
|
|
this.udpCallback = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
start() {
|
|
|
|
const options = this.options;
|
|
|
|
this.reset();
|
|
|
|
|
2018-01-31 07:34:11 +01:00
|
|
|
this.startMillis = Date.now();
|
|
|
|
|
2018-01-31 11:03:13 +01:00
|
|
|
this.attemptTimeoutTimer = setTimeout(() => {
|
|
|
|
this.fatal('timeout');
|
|
|
|
},this.options.attemptTimeout);
|
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
async.series([
|
|
|
|
(c) => {
|
|
|
|
// resolve host names
|
|
|
|
if(!('host' in options)) return c();
|
|
|
|
if(options.host.match(/\d+\.\d+\.\d+\.\d+/)) {
|
|
|
|
options.address = options.host;
|
|
|
|
c();
|
|
|
|
} else {
|
2017-08-09 11:05:55 +02:00
|
|
|
this.parseDns(options.host,c);
|
2017-08-09 12:32:09 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
(c) => {
|
|
|
|
// calculate query port if needed
|
|
|
|
if(!('port_query' in options) && 'port' in options) {
|
|
|
|
const offset = options.port_query_offset || 0;
|
|
|
|
options.port_query = options.port + offset;
|
|
|
|
}
|
|
|
|
c();
|
|
|
|
},
|
|
|
|
(c) => {
|
|
|
|
// run
|
|
|
|
this.run(this.initState());
|
|
|
|
}
|
|
|
|
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
parseDns(host,c) {
|
|
|
|
const resolveStandard = (host,c) => {
|
2017-08-09 12:04:32 +02:00
|
|
|
if(this.debug) console.log("Standard DNS Lookup: " + host);
|
2017-08-09 12:32:09 +02:00
|
|
|
dns.lookup(host, (err,address,family) => {
|
|
|
|
if(err) return this.fatal(err);
|
2017-08-09 12:04:32 +02:00
|
|
|
if(this.debug) console.log(address);
|
2017-08-09 11:05:55 +02:00
|
|
|
this.options.address = address;
|
2017-08-09 12:32:09 +02:00
|
|
|
c();
|
|
|
|
});
|
|
|
|
};
|
2017-08-09 11:05:55 +02:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
const resolveSrv = (srv,host,c) => {
|
2017-08-09 12:04:32 +02:00
|
|
|
if(this.debug) console.log("SRV DNS Lookup: " + srv+'.'+host);
|
2017-08-09 12:32:09 +02:00
|
|
|
dns.resolve(srv+'.'+host, 'SRV', (err,addresses) => {
|
2017-08-09 12:04:32 +02:00
|
|
|
if(this.debug) console.log(err, addresses);
|
2017-08-09 12:32:09 +02:00
|
|
|
if(err) return resolveStandard(host,c);
|
|
|
|
if(addresses.length >= 1) {
|
|
|
|
const line = addresses[0];
|
2017-08-09 11:05:55 +02:00
|
|
|
this.options.port = line.port;
|
|
|
|
const srvhost = line.name;
|
2015-02-21 18:21:56 +01:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
if(srvhost.match(/\d+\.\d+\.\d+\.\d+/)) {
|
2017-08-09 11:05:55 +02:00
|
|
|
this.options.address = srvhost;
|
2017-08-09 12:32:09 +02:00
|
|
|
c();
|
|
|
|
} else {
|
|
|
|
// resolve yet again
|
|
|
|
resolveStandard(srvhost,c);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return resolveStandard(host,c);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
if(this.srvRecord) resolveSrv(this.srvRecord,host,c);
|
|
|
|
else resolveStandard(host,c);
|
|
|
|
}
|
|
|
|
|
|
|
|
// utils
|
|
|
|
/** @returns {Reader} */
|
|
|
|
reader(buffer) {
|
|
|
|
return new Reader(this,buffer);
|
|
|
|
}
|
|
|
|
translate(obj,trans) {
|
|
|
|
for(const from of Object.keys(trans)) {
|
2017-08-09 11:05:55 +02:00
|
|
|
const to = trans[from];
|
2017-08-09 12:32:09 +02:00
|
|
|
if(from in obj) {
|
|
|
|
if(to) obj[to] = obj[from];
|
|
|
|
delete obj[from];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setTimeout(c,t) {
|
|
|
|
if(this.finished) return 0;
|
|
|
|
const id = setTimeout(c,t);
|
|
|
|
this.timers.push(id);
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
trueTest(str) {
|
|
|
|
if(typeof str === 'boolean') return str;
|
|
|
|
if(typeof str === 'number') return str !== 0;
|
|
|
|
if(typeof str === 'string') {
|
|
|
|
if(str.toLowerCase() === 'true') return true;
|
|
|
|
if(str === 'yes') return true;
|
|
|
|
if(str === '1') return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
_tcpConnect(c) {
|
|
|
|
if(this.tcpSocket) return c(this.tcpSocket);
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
let connected = false;
|
|
|
|
let received = Buffer.from([]);
|
|
|
|
const address = this.options.address;
|
2017-08-09 11:05:55 +02:00
|
|
|
const port = this.options.port_query;
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 11:05:55 +02:00
|
|
|
const socket = this.tcpSocket = net.connect(port,address,() => {
|
2017-08-09 12:32:09 +02:00
|
|
|
if(this.debug) console.log(address+':'+port+" TCPCONNECTED");
|
|
|
|
connected = true;
|
|
|
|
c(socket);
|
|
|
|
});
|
|
|
|
socket.setNoDelay(true);
|
|
|
|
if(this.debug) console.log(address+':'+port+" TCPCONNECT");
|
|
|
|
|
|
|
|
const writeHook = socket.write;
|
|
|
|
socket.write = (...args) => {
|
2017-08-10 13:49:42 +02:00
|
|
|
if(this.debug) {
|
|
|
|
console.log(address+':'+port+" TCP-->");
|
|
|
|
console.log(HexUtil.debugDump(args[0]));
|
|
|
|
}
|
2017-08-09 12:32:09 +02:00
|
|
|
writeHook.apply(socket,args);
|
|
|
|
};
|
|
|
|
|
|
|
|
socket.on('error', () => {});
|
|
|
|
socket.on('close', () => {
|
|
|
|
if(!this.tcpCallback) return;
|
|
|
|
if(connected) return this.fatal('Socket closed while waiting on TCP');
|
|
|
|
else return this.fatal('TCP Connection Refused');
|
|
|
|
});
|
|
|
|
socket.on('data', (data) => {
|
|
|
|
if(!this.tcpCallback) return;
|
2017-08-10 13:49:42 +02:00
|
|
|
if(this.debug) {
|
|
|
|
console.log(address+':'+port+" <--TCP");
|
|
|
|
console.log(HexUtil.debugDump(data));
|
|
|
|
}
|
2017-08-09 12:32:09 +02:00
|
|
|
received = Buffer.concat([received,data]);
|
|
|
|
if(this.tcpCallback(received)) {
|
|
|
|
clearTimeout(this.tcpTimeoutTimer);
|
2017-08-09 11:05:55 +02:00
|
|
|
this.tcpCallback = false;
|
2017-08-09 12:32:09 +02:00
|
|
|
received = Buffer.from([]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
tcpSend(buffer,ondata) {
|
|
|
|
process.nextTick(() => {
|
|
|
|
if(this.tcpCallback) return this.fatal('Attempted to send TCP packet while still waiting on a managed response');
|
2017-08-09 11:05:55 +02:00
|
|
|
this._tcpConnect((socket) => {
|
2017-08-09 12:32:09 +02:00
|
|
|
socket.write(buffer);
|
|
|
|
});
|
|
|
|
if(!ondata) return;
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 11:05:55 +02:00
|
|
|
this.tcpTimeoutTimer = this.setTimeout(() => {
|
|
|
|
this.tcpCallback = false;
|
|
|
|
this.fatal('TCP Watchdog Timeout');
|
2018-01-31 11:03:13 +01:00
|
|
|
},this.options.socketTimeout);
|
2017-08-09 11:05:55 +02:00
|
|
|
this.tcpCallback = ondata;
|
2017-08-09 12:32:09 +02:00
|
|
|
});
|
|
|
|
}
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 12:32:09 +02:00
|
|
|
udpSend(buffer,onpacket,ontimeout) {
|
|
|
|
process.nextTick(() => {
|
|
|
|
if(this.udpCallback) return this.fatal('Attempted to send UDP packet while still waiting on a managed response');
|
2017-08-09 11:05:55 +02:00
|
|
|
this._udpSendNow(buffer);
|
2017-08-09 12:32:09 +02:00
|
|
|
if(!onpacket) return;
|
2014-10-29 08:02:03 +01:00
|
|
|
|
2017-08-09 11:05:55 +02:00
|
|
|
this.udpTimeoutTimer = this.setTimeout(() => {
|
|
|
|
this.udpCallback = false;
|
2017-08-09 12:32:09 +02:00
|
|
|
let timeout = false;
|
|
|
|
if(!ontimeout || ontimeout() !== true) timeout = true;
|
|
|
|
if(timeout) this.fatal('UDP Watchdog Timeout');
|
2018-01-31 11:03:13 +01:00
|
|
|
},this.options.socketTimeout);
|
2017-08-09 11:05:55 +02:00
|
|
|
this.udpCallback = onpacket;
|
2017-08-09 12:32:09 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
_udpSendNow(buffer) {
|
|
|
|
if(!('port_query' in this.options)) return this.fatal('Attempted to send without setting a port');
|
|
|
|
if(!('address' in this.options)) return this.fatal('Attempted to send without setting an address');
|
|
|
|
|
|
|
|
if(typeof buffer === 'string') buffer = Buffer.from(buffer,'binary');
|
|
|
|
|
2017-08-10 13:49:42 +02:00
|
|
|
if(this.debug) {
|
|
|
|
console.log(this.options.address+':'+this.options.port_query+" UDP-->");
|
|
|
|
console.log(HexUtil.debugDump(buffer));
|
|
|
|
}
|
2017-08-09 12:32:09 +02:00
|
|
|
this.udpSocket.send(buffer,0,buffer.length,this.options.port_query,this.options.address);
|
|
|
|
}
|
|
|
|
_udpResponse(buffer) {
|
|
|
|
if(this.udpCallback) {
|
|
|
|
const result = this.udpCallback(buffer);
|
|
|
|
if(result === true) {
|
|
|
|
// we're done with this udp session
|
|
|
|
clearTimeout(this.udpTimeoutTimer);
|
|
|
|
this.udpCallback = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.udpResponse(buffer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
udpResponse() {}
|
2017-08-09 11:05:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = Core;
|