diff --git a/app.js b/app.js index e8ddd73a..66f082e1 100644 --- a/app.js +++ b/app.js @@ -1,4 +1,5 @@ var express = require('express') +var fs = require('fs') var bodyParser = require('body-parser') var morgan = require('morgan') var path = require('path') @@ -11,7 +12,7 @@ var webpackConfig = require('./webpack.config') var setupBasicAuth = require('./lib/setup-basic-auth') var Manager = require('./lib/manager') var Missions = require('./lib/missions') -var Mods = require('./lib/mods') +var SteamMods = require('./lib/steam_mods') var Logs = require('./lib/logs') var Settings = require('./lib/settings') @@ -29,17 +30,44 @@ app.use(morgan(config.logFormat || 'dev')) app.use(serveStatic(path.join(__dirname, 'public'))) -var logs = new Logs(config) +/* +Workaround for Steam Workshop with Linux Arma server + +Absolute paths are not supported. +Create symlink in Arma folder to Workshop folder. +Rewrite Workshop mods to use relative path to symlinked folder instead. +*/ +if (config.type === 'linux' && config.steam && config.steam.path) { + var tempWorkshopFolder = path.join(config.path, 'workshop') + try { + var stat = fs.lstatSync(tempWorkshopFolder) + if (!stat.isSymbolicLink()) { + console.error('Please remove workshop folder from Arma directory manually and restart application') + process.exit(1) + } + fs.unlinkSync(tempWorkshopFolder) + } catch (err) { + if (err.code !== 'ENOENT') { + console.error('Something went wrong when creating workaround for workshop') + console.error(err) + process.exit(1) + } + } + + fs.symlinkSync(config.steam.path, tempWorkshopFolder) +} -var manager = new Manager(config, logs) -manager.load() +var logs = new Logs(config) var missions = new Missions(config) -var mods = new Mods(config) +var mods = new SteamMods(config) mods.updateMods() var settings = new Settings(config) +var manager = new Manager(config, logs, mods) +manager.load() + app.use('/api/logs', require('./routes/logs')(logs)) app.use('/api/missions', require('./routes/missions')(missions)) app.use('/api/mods', require('./routes/mods')(mods)) diff --git a/config.js.example b/config.js.example index 2731db86..56667173 100644 --- a/config.js.example +++ b/config.js.example @@ -13,6 +13,12 @@ module.exports = { '@mod1', '@mod2', ], + steam: { + apiKey: '1234567890ABCDE', // https://steamcommunity.com/dev/apikey + path: 'path-to-main-steam-folder', + username: 'steam_username', + password: 'steam_password', + }, admins: [], // add steam IDs here to enable #login without password auth: { // If both username and password is set, HTTP Basic Auth will be used. You may use an array to specify more than one user. username: '', // Username for HTTP Basic Auth diff --git a/lib/manager.js b/lib/manager.js index b7edca7f..132727b8 100644 --- a/lib/manager.js +++ b/lib/manager.js @@ -5,9 +5,10 @@ var Server = require('./server') var filePath = 'servers.json' -var Manager = function (config, logs) { +var Manager = function (config, logs, mods) { this.config = config this.logs = logs + this.mods = mods this.serversArr = [] this.serversHash = {} } @@ -41,7 +42,7 @@ Manager.prototype.removeServer = function (id) { } Manager.prototype._addServer = function (data) { - var server = new Server(this.config, this.logs, data) + var server = new Server(this.config, this.logs, this.mods, data) this.serversArr.push(server) this.serversArr.sort(function (a, b) { return a.title.localeCompare(b.title) diff --git a/lib/mods.js b/lib/mods.js index 3ee34a2a..0395a3dd 100644 --- a/lib/mods.js +++ b/lib/mods.js @@ -21,6 +21,12 @@ Mods.prototype.delete = function (mod, cb) { }) } +Mods.prototype.find = function (id) { + this.mods.find(function (mod) { + return mod.id === id + }) +} + Mods.prototype.updateMods = function () { var self = this glob('**/{@*,csla,gm,vn}/addons', { cwd: self.config.path }, function (err, files) { @@ -28,9 +34,11 @@ Mods.prototype.updateMods = function () { console.log(err) } else { var mods = files.map(function (file) { + var name = path.join(file, '..') return { // Find actual parent mod folder from addons folder - name: path.join(file, '..') + id: name, + name: name } }) diff --git a/lib/server.js b/lib/server.js index 7c785a0e..24cb90a8 100644 --- a/lib/server.js +++ b/lib/server.js @@ -17,9 +17,10 @@ var queryTypes = { ofpresistance: 'operationflashpoint' } -var Server = function (config, logs, options) { +var Server = function (config, logs, mods, options) { this.config = config this.logs = logs + this.modsManager = mods this.update(options) } @@ -95,6 +96,21 @@ Server.prototype.queryStatus = function () { ) } +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) { + if (self.config.type === 'linux' && self.config.steam && self.config.steam.path) { + return mod.path.replace(self.config.steam.path, 'workshop/') + } + + return mod.path + }) +} + Server.prototype.getParameters = function () { var parameters = [] @@ -132,6 +148,7 @@ Server.prototype.start = function () { return this } + var mods = this.getMods() var parameters = this.getParameters() var server = new ArmaServer.Server({ additionalConfigurationOptions: this.getAdditionalConfigurationOptions(), @@ -147,7 +164,7 @@ Server.prototype.start = function () { hostname: this.createServerTitle(this.title), localClient: this.number_of_headless_clients > 0 ? ['127.0.0.1'] : null, missions: this.missions, - mods: this.mods, + mods: mods, motd: (this.motd && this.motd.split('\n')) || null, parameters: parameters, password: this.password, @@ -197,6 +214,7 @@ Server.prototype.startHeadlessClientsIfNeeded = function () { } Server.prototype.startHeadlessClients = function () { + var mods = this.getMods() var parameters = this.getParameters() var self = this var headlessClientInstances = _.times(this.number_of_headless_clients, function (i) { @@ -204,7 +222,7 @@ Server.prototype.startHeadlessClients = function () { filePatching: self.file_patching, game: self.config.game, host: '127.0.0.1', - mods: self.mods, + mods: mods, parameters: parameters, password: self.password, path: self.config.path, diff --git a/lib/steam_mods.js b/lib/steam_mods.js new file mode 100644 index 00000000..dc3db6ec --- /dev/null +++ b/lib/steam_mods.js @@ -0,0 +1,61 @@ +var events = require('events') +var ArmaSteamWorkshop = require('arma-steam-workshop') + +var SteamMods = function (config) { + this.config = config + this.armaSteamWorkshop = new ArmaSteamWorkshop(this.config.steam) + this.mods = [] +} + +SteamMods.prototype = new events.EventEmitter() + +SteamMods.prototype.delete = function (mod, cb) { + var self = this + this.armaSteamWorkshop.deleteMod(mod, function (err) { + if (err) { + console.log(err) + } else { + self.updateMods() + } + + if (cb) { + cb(err) + } + }) +} + +SteamMods.prototype.find = function (id) { + return this.mods.find(function (mod) { + return mod.id === id + }) +} + +SteamMods.prototype.download = function (workshopId, cb) { + var self = this + this.armaSteamWorkshop.downloadMod(workshopId, function (err) { + self.updateMods() + + if (cb) { + cb(err) + } + }) + self.updateMods() +} + +SteamMods.prototype.search = function (query, cb) { + this.armaSteamWorkshop.search(query, cb) +} + +SteamMods.prototype.updateMods = function () { + var self = this + this.armaSteamWorkshop.mods(function (err, mods) { + if (err) { + console.log(err) + } else { + self.mods = mods + self.emit('mods', mods) + } + }) +} + +module.exports = SteamMods diff --git a/package.json b/package.json index f4f294bc..aa2edebc 100644 --- a/package.json +++ b/package.json @@ -18,6 +18,7 @@ }, "dependencies": { "arma-server": "0.0.10", + "arma-steam-workshop": "https://github.com/Dahlgren/node-arma-steam-workshop/tarball/57f11ee85ccac0a18a74c5acb5645c9af78ec5c0", "async": "^0.9.0", "backbone": "1.3.3", "backbone.bootstrap-modal": "https://github.com/powmedia/backbone.bootstrap-modal/archive/632210077c2424be2ee6ea2aafe0d3fe016ae524.tar.gz", diff --git a/public/js/app/models/mod.js b/public/js/app/models/mod.js index f5a278f5..ea3b05c3 100644 --- a/public/js/app/models/mod.js +++ b/public/js/app/models/mod.js @@ -2,8 +2,11 @@ var Backbone = require('backbone') module.exports = Backbone.Model.extend({ defaults: { - name: '' + id: '', + downloading: false, + name: '', + path: '' }, - idAttribute: 'name', + idAttribute: 'id', urlRoot: '/api/mods/' }) diff --git a/public/js/app/views/mods/download/form.js b/public/js/app/views/mods/download/form.js new file mode 100644 index 00000000..05d12448 --- /dev/null +++ b/public/js/app/views/mods/download/form.js @@ -0,0 +1,82 @@ +var $ = require('jquery') +var _ = require('underscore') +var Marionette = require('marionette') +var Ladda = require('ladda') +var Mods = require('app/collections/mods') +var tpl = require('tpl/mods/download/form.html') +var sweetAlert = require('sweet-alert') + +module.exports = Marionette.ItemView.extend({ + + events: { + submit: 'beforeSubmit' + }, + + template: _.template(tpl), + + initialize: function (options) { + this.mods = options.mods + this.collection = new Mods() + this.bind('ok', this.submit) + this.bind('shown', this.shown) + + var self = this + this.listenTo(this.mods, 'change reset add remove', function () { + self.collection.trigger('reset') + }) + }, + + beforeSubmit: function (e) { + e.preventDefault() + this.submit() + }, + + shown: function (modal) { + var $okBtn = modal.$el.find('.btn.ok') + $okBtn.addClass('ladda-button').attr('data-style', 'expand-left') + + this.laddaBtn = Ladda.create($okBtn.get(0)) + + this.$el.find('form .query').focus() + }, + + submit: function (modal) { + var self = this + var $form = this.$el.find('form') + + if (modal) { + self.modal.preventClose() + } + + $form.find('.form-group').removeClass('has-error') + $form.find('.help-block').text('') + + this.laddaBtn.start() + self.modal.$el.find('.btn.cancel').addClass('disabled') + + $.ajax({ + url: '/api/mods/', + type: 'POST', + data: { + id: $form.find('.query').val() + }, + dataType: 'json', + success: function (resp) { + self.laddaBtn.stop() + self.modal.$el.find('.ladda-button').removeClass('disabled') + self.modal.close() + }, + error: function (resp) { + self.laddaBtn.stop() + self.modal.$el.find('.ladda-button').removeClass('disabled') + self.modal.close() + + sweetAlert({ + title: 'Error', + text: 'An error occurred, please consult the logs', + type: 'error' + }) + } + }) + } +}) diff --git a/public/js/app/views/mods/list.js b/public/js/app/views/mods/list.js index a752ce35..91549c97 100644 --- a/public/js/app/views/mods/list.js +++ b/public/js/app/views/mods/list.js @@ -1,8 +1,11 @@ var $ = require('jquery') var _ = require('underscore') var Marionette = require('marionette') +var BootstrapModal = require('backbone.bootstrap-modal') var ListItemView = require('app/views/mods/list_item') +var DownloadFormView = require('app/views/mods/download/form') +var SearchFormView = require('app/views/mods/search/form') var tpl = require('tpl/mods/list.html') var template = _.template(tpl) @@ -13,7 +16,22 @@ module.exports = Marionette.CompositeView.extend({ template: template, events: { - 'click #refresh': 'refresh' + 'click #download': 'download', + 'click #refresh': 'refresh', + 'click #search': 'search' + }, + + download: function (event) { + event.preventDefault() + var view = new DownloadFormView({ mods: this.collection }) + var modal = new BootstrapModal({ + content: view, + animate: true, + cancelText: 'Close', + okText: 'Download' + }) + view.modal = modal + modal.open() }, refresh: function (event) { @@ -28,5 +46,18 @@ module.exports = Marionette.CompositeView.extend({ } }) + }, + + search: function (event) { + event.preventDefault() + var view = new SearchFormView({ mods: this.collection }) + var modal = new BootstrapModal({ + content: view, + animate: true, + cancelText: 'Close', + okText: 'Search' + }) + view.modal = modal + modal.open() } }) diff --git a/public/js/app/views/mods/list_item.js b/public/js/app/views/mods/list_item.js index 724e3a56..df33946f 100644 --- a/public/js/app/views/mods/list_item.js +++ b/public/js/app/views/mods/list_item.js @@ -1,4 +1,6 @@ +var $ = require('jquery') var _ = require('underscore') +var Ladda = require('ladda') var Marionette = require('marionette') var sweetAlert = require('sweet-alert') @@ -11,9 +13,40 @@ module.exports = Marionette.ItemView.extend({ template: template, events: { + 'click .install': 'installMod', 'click .destroy': 'deleteMod' }, + modelEvents: { + change: 'render' + }, + + installMod: function (event) { + var self = this + event.preventDefault() + + this.laddaBtn = Ladda.create(this.$el.find('.ladda-button').get(0)) + this.laddaBtn.start() + this.$el.find('.ladda-button').addClass('disabled') + + $.ajax({ + url: '/api/mods/', + type: 'POST', + data: { + id: this.model.get('id') + }, + dataType: 'json', + success: function (resp) { + self.laddaBtn.stop() + self.$el.find('.ladda-button').removeClass('disabled') + }, + error: function (resp) { + self.laddaBtn.stop() + self.$el.find('.ladda-button').removeClass('disabled') + } + }) + }, + deleteMod: function (event) { var self = this sweetAlert({ diff --git a/public/js/app/views/mods/search/form.js b/public/js/app/views/mods/search/form.js new file mode 100644 index 00000000..9708f828 --- /dev/null +++ b/public/js/app/views/mods/search/form.js @@ -0,0 +1,90 @@ +var $ = require('jquery') +var _ = require('underscore') +var Marionette = require('marionette') +var ListItemView = require('app/views/mods/search/list_item') +var Ladda = require('ladda') +var Mods = require('app/collections/mods') +var tpl = require('tpl/mods/search/form.html') +var sweetAlert = require('sweet-alert') + +module.exports = Marionette.CompositeView.extend({ + + events: { + submit: 'beforeSubmit' + }, + + childView: ListItemView, + childViewContainer: 'tbody', + template: _.template(tpl), + + initialize: function (options) { + this.mods = options.mods + this.collection = new Mods() + this.bind('ok', this.submit) + this.bind('shown', this.shown) + + var self = this + this.listenTo(this.mods, 'change reset add remove', function () { + self.collection.trigger('reset') + }) + }, + + childViewOptions: function (options) { + options.set('mods', this.mods) + }, + + beforeSubmit: function (e) { + e.preventDefault() + this.submit() + }, + + shown: function (modal) { + var $okBtn = modal.$el.find('.btn.ok') + $okBtn.addClass('ladda-button').attr('data-style', 'expand-left') + + this.laddaBtn = Ladda.create($okBtn.get(0)) + + this.$el.find('form .query').focus() + }, + + submit: function (modal) { + var self = this + var $form = this.$el.find('form') + + if (modal) { + self.modal.preventClose() + } + + $form.find('.form-group').removeClass('has-error') + $form.find('.help-block').text('') + + this.laddaBtn.start() + self.modal.$el.find('.btn.cancel').addClass('disabled') + + $.ajax({ + url: '/api/mods/search', + type: 'POST', + data: { + query: $form.find('.query').val() + }, + dataType: 'json', + success: function (data) { + self.laddaBtn.stop() + self.modal.$el.find('.btn.cancel').removeClass('disabled') + self.collection.set(data) + }, + error: function () { + self.laddaBtn.stop() + $form.find('.form-group').addClass('has-error') + $form.find('.help-block').text('Problem searching, try again') + self.modal.$el.find('.btn.cancel').removeClass('disabled') + + sweetAlert({ + title: 'Error', + text: 'An error occurred, please consult the logs', + type: 'error' + }) + } + }) + } +}) diff --git a/public/js/app/views/mods/search/list_item.js b/public/js/app/views/mods/search/list_item.js new file mode 100644 index 00000000..4bdcdafb --- /dev/null +++ b/public/js/app/views/mods/search/list_item.js @@ -0,0 +1,52 @@ +var $ = require('jquery') +var _ = require('underscore') +var Marionette = require('marionette') +var Ladda = require('ladda') +var tpl = require('tpl/mods/search/list_item.html') + +var template = _.template(tpl) + +module.exports = Marionette.ItemView.extend({ + tagName: 'tr', + template: template, + + events: { + 'click .install': 'install' + }, + + templateHelpers: { + downloading: function () { + if (this.mods.get(this.id)) { + return this.mods.get(this.id).get('downloading') + } + + return false + } + }, + + install: function (event) { + var self = this + event.preventDefault() + + this.laddaBtn = Ladda.create(this.$el.find('.ladda-button').get(0)) + this.laddaBtn.start() + this.$el.find('.ladda-button').addClass('disabled') + + $.ajax({ + url: '/api/mods/', + type: 'POST', + data: { + id: this.model.get('id') + }, + dataType: 'json', + success: function (resp) { + self.laddaBtn.stop() + self.$el.find('.ladda-button').removeClass('disabled') + }, + error: function (resp) { + self.laddaBtn.stop() + self.$el.find('.ladda-button').removeClass('disabled') + } + }) + } +}) diff --git a/public/js/app/views/servers/info.js b/public/js/app/views/servers/info.js index fb5de699..b6169c06 100644 --- a/public/js/app/views/servers/info.js +++ b/public/js/app/views/servers/info.js @@ -57,5 +57,16 @@ module.exports = Marionette.LayoutView.extend({ self.render() }) }) + }, + + templateHelpers: function () { + var self = this + return { + mods: self.options.mods.filter(function (mod) { + return self.model.get('mods').indexOf(mod.get('id')) >= 0 + }).map(function (mod) { + return mod.get('name') + }) + } } }) diff --git a/public/js/app/views/servers/mods/list_item.js b/public/js/app/views/servers/mods/list_item.js index 31686723..923be895 100644 --- a/public/js/app/views/servers/mods/list_item.js +++ b/public/js/app/views/servers/mods/list_item.js @@ -11,7 +11,7 @@ module.exports = ModListItemView.extend({ templateHelpers: function () { return { - enabled: this.options.server.get('mods').indexOf(this.model.get('name')) > -1 + enabled: this.options.server.get('mods').indexOf(this.model.get('id')) > -1 } } }) diff --git a/public/js/app/views/servers/view.js b/public/js/app/views/servers/view.js index d93123aa..1e6dc511 100644 --- a/public/js/app/views/servers/view.js +++ b/public/js/app/views/servers/view.js @@ -39,7 +39,7 @@ module.exports = Marionette.LayoutView.extend({ }, onRender: function () { - this.infoView.show(new InfoView({ model: this.model })) + this.infoView.show(new InfoView({ model: this.model, mods: this.mods })) this.missionsView.show(new MissionsView({ missions: this.missions, model: this.model })) this.modsView.show(new ModsListView({ collection: this.mods, server: this.model })) this.parametersView.show(new ParametersListView({ model: this.model })) diff --git a/public/js/tpl/mods/download/form.html b/public/js/tpl/mods/download/form.html new file mode 100644 index 00000000..e763533e --- /dev/null +++ b/public/js/tpl/mods/download/form.html @@ -0,0 +1,11 @@ +
+ Install mod from Steam Workshop by id +
+ + diff --git a/public/js/tpl/mods/list.html b/public/js/tpl/mods/list.html index bc651351..ca874cc6 100644 --- a/public/js/tpl/mods/list.html +++ b/public/js/tpl/mods/list.html @@ -3,6 +3,16 @@ Refresh + + + Download specific mod + + + + + Search & Download + +
-
-
- Delete
-
+ <% if (downloading) { %>
+
+
+
+ <% } else { %>
+
+ <% if (needsUpdate) { %>
+
+
+ Update
+
+ <% } %>
-
+
+
+ Delete
+
+
+
+
+ <% } %>
|
diff --git a/public/js/tpl/mods/search/form.html b/public/js/tpl/mods/search/form.html
new file mode 100644
index 00000000..79ab0a57
--- /dev/null
+++ b/public/js/tpl/mods/search/form.html
@@ -0,0 +1,23 @@
+
Mod | ++ |
---|
+ <%-title%> +
++ <%-description%> +
++ <% if (downloading()) { %> +
+ <%-fileSizeFormatted%> +
+ +