2017-08-24 09:15:06 +00:00
|
|
|
var _ = require('lodash')
|
|
|
|
var events = require('events')
|
|
|
|
var Gamedig = require('gamedig')
|
|
|
|
var slugify = require('slugify')
|
2014-02-13 22:47:51 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
var ArmaServer = require('arma-server')
|
2014-09-12 23:51:31 +00:00
|
|
|
|
2021-06-16 17:04:59 +00:00
|
|
|
var virtualServer = require('./virtualServer')
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
var queryInterval = 5000
|
2015-04-25 10:17:24 +00:00
|
|
|
var queryTypes = {
|
|
|
|
arma1: 'arma',
|
|
|
|
arma2: 'arma2',
|
|
|
|
arma2oa: 'arma2',
|
|
|
|
arma3: 'arma3',
|
2017-03-18 18:33:19 +00:00
|
|
|
arma3_x64: 'arma3',
|
2015-04-25 10:17:24 +00:00
|
|
|
cwa: 'operationflashpoint',
|
|
|
|
ofp: 'operationflashpoint',
|
2017-08-24 09:15:06 +00:00
|
|
|
ofpresistance: 'operationflashpoint'
|
|
|
|
}
|
2014-02-13 22:47:51 +00:00
|
|
|
|
2021-07-10 13:54:10 +00:00
|
|
|
var Server = function (config, logs, mods, options) {
|
2017-08-24 09:15:06 +00:00
|
|
|
this.config = config
|
|
|
|
this.logs = logs
|
2021-07-10 13:54:10 +00:00
|
|
|
this.modsManager = mods
|
2017-08-24 09:15:06 +00:00
|
|
|
this.update(options)
|
|
|
|
}
|
2014-06-04 22:19:54 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype = new events.EventEmitter()
|
2014-02-13 22:47:51 +00:00
|
|
|
|
2021-07-10 13:38:50 +00:00
|
|
|
Server.prototype.createServerTitle = function (title) {
|
|
|
|
if (this.config.prefix) {
|
|
|
|
title = this.config.prefix + title
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.config.suffix) {
|
|
|
|
title = title + this.config.suffix
|
|
|
|
}
|
|
|
|
|
|
|
|
return title
|
|
|
|
}
|
|
|
|
|
2017-03-31 15:08:56 +00:00
|
|
|
Server.prototype.generateId = function () {
|
2017-08-24 09:15:06 +00:00
|
|
|
return slugify(this.title).replace(/\./g, '-')
|
|
|
|
}
|
2017-03-31 15:08:56 +00:00
|
|
|
|
2014-09-12 23:51:31 +00:00
|
|
|
Server.prototype.update = function (options) {
|
2019-05-22 14:21:28 +00:00
|
|
|
this.additionalConfigurationOptions = options.additionalConfigurationOptions
|
2017-08-24 09:15:06 +00:00
|
|
|
this.admin_password = options.admin_password
|
2017-10-01 14:58:30 +00:00
|
|
|
this.allowed_file_patching = options.allowed_file_patching
|
2017-08-24 09:15:06 +00:00
|
|
|
this.auto_start = options.auto_start
|
|
|
|
this.battle_eye = options.battle_eye
|
2017-10-01 14:58:30 +00:00
|
|
|
this.file_patching = options.file_patching
|
2017-09-30 19:49:42 +00:00
|
|
|
this.forcedDifficulty = options.forcedDifficulty || null
|
2017-08-24 09:15:06 +00:00
|
|
|
this.max_players = options.max_players
|
|
|
|
this.missions = options.missions
|
|
|
|
this.mods = options.mods || []
|
2017-09-30 19:49:42 +00:00
|
|
|
this.motd = options.motd || null
|
2017-08-24 09:15:06 +00:00
|
|
|
this.number_of_headless_clients = options.number_of_headless_clients || 0
|
|
|
|
this.password = options.password
|
|
|
|
this.parameters = options.parameters
|
|
|
|
this.persistent = options.persistent
|
|
|
|
this.port = options.port || 2302
|
|
|
|
this.title = options.title
|
|
|
|
this.von = options.von
|
|
|
|
this.verify_signatures = options.verify_signatures
|
|
|
|
|
|
|
|
this.id = this.generateId()
|
|
|
|
this.port = parseInt(this.port, 10) // If port is a string then gamedig fails
|
|
|
|
}
|
|
|
|
|
|
|
|
Server.prototype.queryStatus = function () {
|
2015-10-31 13:35:50 +00:00
|
|
|
if (!this.instance) {
|
2017-08-24 09:15:06 +00:00
|
|
|
return
|
2015-10-31 13:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
var self = this
|
2014-11-09 18:07:00 +00:00
|
|
|
Gamedig.query(
|
2015-04-25 10:17:48 +00:00
|
|
|
{
|
2021-07-10 13:38:50 +00:00
|
|
|
type: queryTypes[this.config.game],
|
2015-04-25 10:17:48 +00:00
|
|
|
host: '127.0.0.1',
|
2017-08-24 09:15:06 +00:00
|
|
|
port: self.port
|
2015-04-25 10:17:48 +00:00
|
|
|
},
|
2017-08-24 09:15:06 +00:00
|
|
|
function (state) {
|
2015-10-31 13:35:50 +00:00
|
|
|
if (!self.instance) {
|
2017-08-24 09:15:06 +00:00
|
|
|
return
|
2015-10-31 13:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
if (state.error) {
|
|
|
|
self.state = null
|
2015-04-25 10:17:48 +00:00
|
|
|
} else {
|
2017-08-24 09:15:06 +00:00
|
|
|
self.state = state
|
2021-07-10 13:10:17 +00:00
|
|
|
self.startHeadlessClientsIfNeeded()
|
2015-04-25 10:17:48 +00:00
|
|
|
}
|
2015-01-21 02:18:45 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
self.emit('state')
|
2015-04-25 10:17:48 +00:00
|
|
|
}
|
2017-08-24 09:15:06 +00:00
|
|
|
)
|
|
|
|
}
|
2014-11-09 18:07:00 +00:00
|
|
|
|
2021-07-10 13:54:10 +00:00
|
|
|
Server.prototype.getMods = function () {
|
|
|
|
var self = this
|
|
|
|
return this.mods.map(function (mod) {
|
|
|
|
return self.modsManager.find(mod)
|
|
|
|
}).filter(function (mod) {
|
|
|
|
return mod
|
|
|
|
}).map(function (mod) {
|
|
|
|
return mod.path
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype.getParameters = function () {
|
|
|
|
var parameters = []
|
2016-11-19 22:31:38 +00:00
|
|
|
|
2021-07-10 13:38:50 +00:00
|
|
|
if (this.config.parameters && Array.isArray(this.config.parameters)) {
|
|
|
|
parameters = parameters.concat(this.config.parameters)
|
2016-11-19 22:31:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.parameters && Array.isArray(this.parameters)) {
|
2017-08-24 09:15:06 +00:00
|
|
|
parameters = parameters.concat(this.parameters)
|
2016-11-19 22:31:38 +00:00
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
return parameters
|
2017-09-11 09:28:47 +00:00
|
|
|
}
|
|
|
|
|
2019-05-22 14:21:28 +00:00
|
|
|
Server.prototype.getAdditionalConfigurationOptions = function () {
|
|
|
|
var additionalConfigurationOptions = ''
|
|
|
|
|
2021-07-10 13:38:50 +00:00
|
|
|
if (this.config.additionalConfigurationOptions) {
|
|
|
|
additionalConfigurationOptions += this.config.additionalConfigurationOptions
|
2019-05-22 14:21:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.additionalConfigurationOptions) {
|
|
|
|
if (additionalConfigurationOptions) {
|
|
|
|
additionalConfigurationOptions += '\n'
|
|
|
|
}
|
|
|
|
|
|
|
|
additionalConfigurationOptions += this.additionalConfigurationOptions
|
|
|
|
}
|
|
|
|
|
|
|
|
return additionalConfigurationOptions
|
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype.start = function () {
|
2018-01-02 13:30:39 +00:00
|
|
|
if (this.instance) {
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
|
2021-06-16 17:04:59 +00:00
|
|
|
var self = this
|
2021-07-10 14:03:32 +00:00
|
|
|
var mods = this.getMods()
|
2021-06-16 17:04:59 +00:00
|
|
|
|
|
|
|
if (self.config.virtualServer && self.config.virtualServer.enabled) {
|
2021-07-10 14:03:32 +00:00
|
|
|
virtualServer.create(self.config, mods)
|
2021-06-16 17:04:59 +00:00
|
|
|
.then((serverFolder) => {
|
|
|
|
self.virtualServerFolder = serverFolder
|
|
|
|
self.path = serverFolder
|
|
|
|
self.realStart()
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
console.error('Error creating virtual server folder:', err)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
self.path = self.config.path
|
|
|
|
self.realStart()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Server.prototype.realStart = function () {
|
|
|
|
if (this.instance) {
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
var parameters = this.getParameters()
|
2015-02-13 16:13:55 +00:00
|
|
|
var server = new ArmaServer.Server({
|
2019-05-22 14:21:28 +00:00
|
|
|
additionalConfigurationOptions: this.getAdditionalConfigurationOptions(),
|
2021-07-10 13:38:50 +00:00
|
|
|
admins: this.config.admins,
|
2017-10-01 14:58:30 +00:00
|
|
|
allowedFilePatching: this.allowed_file_patching || 1,
|
2014-09-12 23:51:31 +00:00
|
|
|
battleEye: this.battle_eye ? 1 : 0,
|
|
|
|
config: this.id,
|
2015-01-16 21:50:31 +00:00
|
|
|
disableVoN: this.von ? 0 : 1,
|
2021-07-10 13:38:50 +00:00
|
|
|
game: this.config.game,
|
2017-10-01 14:58:30 +00:00
|
|
|
filePatching: this.file_patching || false,
|
2017-09-30 19:49:42 +00:00
|
|
|
forcedDifficulty: this.forcedDifficulty || null,
|
2017-08-24 09:15:06 +00:00
|
|
|
headlessClients: this.number_of_headless_clients > 0 ? ['127.0.0.1'] : null,
|
2021-07-10 13:38:50 +00:00
|
|
|
hostname: this.createServerTitle(this.title),
|
2017-08-24 09:15:06 +00:00
|
|
|
localClient: this.number_of_headless_clients > 0 ? ['127.0.0.1'] : null,
|
2015-08-23 22:16:16 +00:00
|
|
|
missions: this.missions,
|
2021-07-10 14:03:32 +00:00
|
|
|
mods: this.mods,
|
2017-09-30 19:49:42 +00:00
|
|
|
motd: (this.motd && this.motd.split('\n')) || null,
|
2016-11-19 22:31:38 +00:00
|
|
|
parameters: parameters,
|
2014-09-12 23:51:31 +00:00
|
|
|
password: this.password,
|
|
|
|
passwordAdmin: this.admin_password,
|
2021-06-16 17:04:59 +00:00
|
|
|
path: this.path,
|
2014-09-12 23:51:31 +00:00
|
|
|
persistent: this.persistent ? 1 : 0,
|
2015-10-31 14:27:16 +00:00
|
|
|
platform: this.config.type,
|
2014-09-12 23:51:31 +00:00
|
|
|
players: this.max_players,
|
|
|
|
port: this.port,
|
2021-07-10 13:38:50 +00:00
|
|
|
serverMods: this.config.serverMods,
|
2017-08-24 09:15:06 +00:00
|
|
|
verifySignatures: this.verify_signatures ? 2 : 0
|
|
|
|
})
|
|
|
|
server.writeServerConfig()
|
|
|
|
var instance = server.start()
|
|
|
|
var self = this
|
2014-02-13 22:47:51 +00:00
|
|
|
|
2014-11-24 00:56:54 +00:00
|
|
|
instance.on('close', function (code) {
|
2017-08-24 09:15:06 +00:00
|
|
|
clearInterval(self.queryStatusInterval)
|
|
|
|
self.state = null
|
|
|
|
self.pid = null
|
|
|
|
self.instance = null
|
2014-11-09 18:07:00 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
self.stopHeadlessClients()
|
2021-06-16 17:04:59 +00:00
|
|
|
.then(() => {
|
|
|
|
if (self.virtualServerFolder) {
|
|
|
|
virtualServer.remove(self.virtualServerFolder)
|
|
|
|
self.virtualServerFolder = null
|
|
|
|
}
|
|
|
|
self.emit('state')
|
|
|
|
})
|
2017-08-24 09:15:06 +00:00
|
|
|
})
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
this.pid = instance.pid
|
|
|
|
this.instance = instance
|
2021-07-10 13:10:17 +00:00
|
|
|
this.headlessClientInstances = []
|
2014-11-09 18:07:00 +00:00
|
|
|
this.queryStatusInterval = setInterval(function () {
|
2017-08-24 09:15:06 +00:00
|
|
|
self.queryStatus()
|
|
|
|
}, queryInterval)
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2021-07-10 13:10:17 +00:00
|
|
|
this.logs.logServerProcess(this.instance, this.id, 'server')
|
|
|
|
this.logs.cleanupOldLogFiles()
|
2020-12-13 13:48:58 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
this.emit('state')
|
2017-09-11 09:28:47 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
return this
|
|
|
|
}
|
2017-09-11 09:28:47 +00:00
|
|
|
|
2021-07-10 13:10:17 +00:00
|
|
|
Server.prototype.startHeadlessClientsIfNeeded = function () {
|
|
|
|
if (this.number_of_headless_clients > 0 && this.headlessClientInstances.length === 0) {
|
|
|
|
this.startHeadlessClients()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype.startHeadlessClients = function () {
|
|
|
|
var parameters = this.getParameters()
|
|
|
|
var self = this
|
2017-09-11 09:28:47 +00:00
|
|
|
var headlessClientInstances = _.times(this.number_of_headless_clients, function (i) {
|
2015-02-13 16:13:55 +00:00
|
|
|
var headless = new ArmaServer.Headless({
|
2017-10-01 14:58:30 +00:00
|
|
|
filePatching: self.file_patching,
|
2021-07-10 13:38:50 +00:00
|
|
|
game: self.config.game,
|
2017-08-24 09:15:06 +00:00
|
|
|
host: '127.0.0.1',
|
2021-07-10 14:03:32 +00:00
|
|
|
mods: self.mods,
|
2016-11-19 22:31:38 +00:00
|
|
|
parameters: parameters,
|
2017-09-11 09:28:47 +00:00
|
|
|
password: self.password,
|
2021-06-16 17:04:59 +00:00
|
|
|
path: self.path,
|
2017-09-11 09:28:47 +00:00
|
|
|
platform: self.config.type,
|
2017-08-24 09:15:06 +00:00
|
|
|
port: self.port
|
|
|
|
})
|
|
|
|
var headlessInstance = headless.start()
|
2021-07-10 13:10:17 +00:00
|
|
|
self.logs.logServerProcess(headlessInstance, self.id, 'hc_' + (i + 1))
|
2017-08-24 09:15:06 +00:00
|
|
|
return headlessInstance
|
|
|
|
})
|
2015-02-13 16:13:55 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
this.headlessClientInstances = headlessClientInstances
|
2017-09-11 09:28:47 +00:00
|
|
|
}
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype.stop = function (cb) {
|
|
|
|
var handled = false
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2014-11-24 00:56:54 +00:00
|
|
|
this.instance.on('close', function (code) {
|
2014-04-06 22:58:17 +00:00
|
|
|
if (!handled) {
|
2017-08-24 09:15:06 +00:00
|
|
|
handled = true
|
2014-06-04 23:02:09 +00:00
|
|
|
|
|
|
|
if (cb) {
|
2017-08-24 09:15:06 +00:00
|
|
|
cb()
|
2014-06-04 23:02:09 +00:00
|
|
|
}
|
2014-04-06 22:58:17 +00:00
|
|
|
}
|
2017-08-24 09:15:06 +00:00
|
|
|
})
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
this.instance.kill()
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
setTimeout(function () {
|
2014-04-06 22:58:17 +00:00
|
|
|
if (!handled) {
|
2017-08-24 09:15:06 +00:00
|
|
|
handled = true
|
2014-06-04 23:02:09 +00:00
|
|
|
|
|
|
|
if (cb) {
|
2017-08-24 09:15:06 +00:00
|
|
|
cb()
|
2014-06-04 23:02:09 +00:00
|
|
|
}
|
2014-04-06 22:58:17 +00:00
|
|
|
}
|
2017-08-24 09:15:06 +00:00
|
|
|
}, 5000)
|
2014-04-06 22:58:17 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
return this
|
|
|
|
}
|
2014-06-04 22:17:18 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
Server.prototype.stopHeadlessClients = function () {
|
2021-06-16 17:04:59 +00:00
|
|
|
var self = this
|
|
|
|
return Promise.all(this.headlessClientInstances.map(function (headlessClientInstance) {
|
|
|
|
var handled = false
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
|
|
headlessClientInstance.on('close', function () {
|
|
|
|
if (!handled) {
|
|
|
|
handled = true
|
|
|
|
resolve()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
setTimeout(function () {
|
|
|
|
if (!handled) {
|
|
|
|
handled = true
|
|
|
|
resolve()
|
|
|
|
}
|
|
|
|
}, 5000)
|
|
|
|
|
|
|
|
headlessClientInstance.kill()
|
|
|
|
})
|
|
|
|
})).then(function () {
|
|
|
|
self.headlessClientInstances = []
|
2017-08-24 09:15:06 +00:00
|
|
|
})
|
2017-09-11 09:28:47 +00:00
|
|
|
}
|
|
|
|
|
2014-06-04 22:17:18 +00:00
|
|
|
Server.prototype.toJSON = function () {
|
|
|
|
return {
|
2019-05-22 14:21:28 +00:00
|
|
|
additionalConfigurationOptions: this.additionalConfigurationOptions,
|
2014-09-12 23:51:31 +00:00
|
|
|
admin_password: this.admin_password,
|
2017-10-01 14:58:30 +00:00
|
|
|
allowed_file_patching: this.allowed_file_patching,
|
2017-02-20 08:44:07 +00:00
|
|
|
auto_start: this.auto_start,
|
2014-09-12 23:51:31 +00:00
|
|
|
battle_eye: this.battle_eye,
|
2014-06-04 22:17:18 +00:00
|
|
|
id: this.id,
|
2017-10-03 08:56:58 +00:00
|
|
|
file_patching: this.file_patching,
|
2017-09-30 19:49:42 +00:00
|
|
|
forcedDifficulty: this.forcedDifficulty,
|
2014-09-12 23:51:31 +00:00
|
|
|
max_players: this.max_players,
|
2015-08-23 22:16:16 +00:00
|
|
|
missions: this.missions,
|
2017-09-30 19:49:42 +00:00
|
|
|
motd: this.motd,
|
2014-06-04 22:17:18 +00:00
|
|
|
mods: this.mods,
|
2017-09-11 09:28:47 +00:00
|
|
|
number_of_headless_clients: this.number_of_headless_clients,
|
2016-11-19 22:31:38 +00:00
|
|
|
parameters: this.parameters,
|
2014-09-12 23:51:31 +00:00
|
|
|
password: this.password,
|
|
|
|
persistent: this.persistent,
|
2014-06-04 22:17:18 +00:00
|
|
|
pid: this.pid,
|
2014-09-12 23:51:31 +00:00
|
|
|
port: this.port,
|
2014-11-09 18:07:00 +00:00
|
|
|
state: this.state,
|
2014-09-12 23:51:31 +00:00
|
|
|
title: this.title,
|
|
|
|
von: this.von,
|
2017-08-24 09:15:06 +00:00
|
|
|
verify_signatures: this.verify_signatures
|
|
|
|
}
|
|
|
|
}
|
2014-02-13 22:47:51 +00:00
|
|
|
|
2017-08-24 09:15:06 +00:00
|
|
|
module.exports = Server
|