diff options
-rw-r--r-- | module/api/ConfigApi.py | 35 | ||||
-rw-r--r-- | module/config/ConfigManager.py | 12 | ||||
-rw-r--r-- | module/config/ConfigParser.py | 4 | ||||
-rw-r--r-- | module/remote/apitypes.py | 6 | ||||
-rw-r--r-- | module/remote/apitypes_debug.py | 3 | ||||
-rw-r--r-- | module/remote/pyload.thrift | 5 | ||||
-rw-r--r-- | module/web/static/js/models/ConfigHolder.js | 44 | ||||
-rw-r--r-- | module/web/static/js/models/ConfigItem.js | 22 | ||||
-rw-r--r-- | module/web/static/js/views/headerView.js | 3 | ||||
-rw-r--r-- | module/web/static/js/views/queryModal.js | 1 | ||||
-rw-r--r-- | module/web/static/js/views/settingsView.js | 89 | ||||
-rw-r--r-- | module/web/templates/default/settings.html | 148 | ||||
-rw-r--r-- | tests/manager/test_configManager.py | 4 | ||||
-rw-r--r-- | tests/other/test_configparser.py | 3 |
14 files changed, 233 insertions, 146 deletions
diff --git a/module/api/ConfigApi.py b/module/api/ConfigApi.py index 9df9455a2..4fba0c34e 100644 --- a/module/api/ConfigApi.py +++ b/module/api/ConfigApi.py @@ -6,6 +6,14 @@ from module.utils import to_string from ApiComponent import ApiComponent +# helper function to create a ConfigHolder +def toConfigHolder(section, config, values): + holder = ConfigHolder(section, config.name, config.description, config.long_desc) + holder.items = [ConfigItem(option, x.name, x.description, x.type, to_string(x.default), + to_string(values.get(option, x.default))) for option, x in + config.config.iteritems()] + return holder + class ConfigApi(ApiComponent): """ Everything related to configuration """ @@ -39,11 +47,7 @@ class ConfigApi(ApiComponent): """ data = {} for section, config, values in self.core.config.iterCoreSections(): - holder = ConfigHolder(section, config.name, config.description, config.long_desc) - holder.items = [ConfigItem(option, x.name, x.description, x.type, to_string(x.default), - to_string(values.get(option, x.default))) for option, x in config.config.iteritems()] - - data[section] = holder + data[section] = toConfigHolder(section, config, values) return data def getCoreConfig(self): @@ -65,9 +69,9 @@ class ConfigApi(ApiComponent): for name, config, values in self.core.config.iterSections(self.user): if not values: continue item = ConfigInfo(name, config.name, config.description, - self.core.pluginManager.getCategory(name), - self.core.pluginManager.isUserPlugin(name), - values.get("activated", False)) + self.core.pluginManager.getCategory(name), + self.core.pluginManager.isUserPlugin(name), + values.get("activated", False)) data.append(item) return data @@ -80,19 +84,21 @@ class ConfigApi(ApiComponent): """ # TODO: filter user_context / addons when not allowed return [ConfigInfo(name, config.name, config.description, - self.core.pluginManager.getCategory(name), - self.core.pluginManager.isUserPlugin(name)) + self.core.pluginManager.getCategory(name), + self.core.pluginManager.isUserPlugin(name)) for name, config, values in self.core.config.iterSections(self.user)] @RequirePerm(Permission.Plugins) - def configurePlugin(self, plugin): + def loadConfig(self, name): """Get complete config options for desired section - :param plugin: Name of plugin or config section + :param name: Name of plugin or config section :rtype: ConfigHolder """ + # requires at least plugin permissions, but only admin can load core config + config, values = self.core.config.getSection(name) + return toConfigHolder(name, config, values) - pass @RequirePerm(Permission.Plugins) def saveConfig(self, config): @@ -110,9 +116,6 @@ class ConfigApi(ApiComponent): """ self.core.config.delete(plugin, self.user) - @RequirePerm(Permission.Plugins) - def setConfigHandler(self, plugin, iid, value): - pass if Api.extend(ConfigApi): del ConfigApi
\ No newline at end of file diff --git a/module/config/ConfigManager.py b/module/config/ConfigManager.py index 8d908abaf..ff638fd71 100644 --- a/module/config/ConfigManager.py +++ b/module/config/ConfigManager.py @@ -42,6 +42,7 @@ class ConfigManager(ConfigParser): # Entries are saved as (user, section) keys self.values = {} # TODO: similar to a cache, could be deleted periodically + # TODO: user / primaryuid is a bit messy def save(self): self.parser.save() @@ -78,6 +79,8 @@ class ConfigManager(ConfigParser): self.values[user, section] = {} self.core.print_exc() + return self.values[user, section] + @convertKeyError def set(self, section, option, value, sync=True, user=None): """ set config value """ @@ -107,7 +110,7 @@ class ConfigManager(ConfigParser): def delete(self, section, user=False): """ Deletes values saved in db and cached values for given user, NOT meta data Does not trigger an error when nothing was deleted. """ - user = user.primary if user else None + user = primary_uid(user) if (user, section) in self.values: del self.values[user, section] @@ -132,3 +135,10 @@ class ConfigManager(ConfigParser): for name, config in self.config.iteritems(): yield name, config, values[name] if name in values else {} + + def getSection(self, section, user=None): + if section in self.parser and primary_uid(user) is None: + return self.parser.getSection(section) + + values = self.loadValues(section, user) + return self.config.get(section), values diff --git a/module/config/ConfigParser.py b/module/config/ConfigParser.py index 2f974b75e..bf9192270 100644 --- a/module/config/ConfigParser.py +++ b/module/config/ConfigParser.py @@ -168,6 +168,10 @@ class ConfigParser: for name, config in self.config.iteritems(): yield name, config, self.values[name] if name in self.values else {} + def getSection(self, section): + """ Retrieves single config as tuple (section, values) """ + return self.config[section], self.values[section] if section in self.values else {} + def addConfigSection(self, section, name, desc, long_desc, config): """Adds a section to the config. `config` is a list of config tuples as used in plugin api defined as: The order of the config elements is preserved with OrderedDict diff --git a/module/remote/apitypes.py b/module/remote/apitypes.py index e81c960c8..41f9be50e 100644 --- a/module/remote/apitypes.py +++ b/module/remote/apitypes.py @@ -389,8 +389,6 @@ class Iface(object): pass def checkURLs(self, urls): pass - def configurePlugin(self, plugin): - pass def createPackage(self, name, folder, root, password, site, comment, paused): pass def deleteCollLink(self, url): @@ -467,6 +465,8 @@ class Iface(object): pass def isInteractionWaiting(self, mode): pass + def loadConfig(self, name): + pass def login(self, username, password): pass def moveFiles(self, fids, pid): @@ -505,8 +505,6 @@ class Iface(object): pass def searchSuggestions(self, pattern): pass - def setConfigHandler(self, plugin, iid, value): - pass def setConfigValue(self, section, option, value): pass def setInteractionResult(self, iid, result): diff --git a/module/remote/apitypes_debug.py b/module/remote/apitypes_debug.py index 7b1b5e7f3..96673cc99 100644 --- a/module/remote/apitypes_debug.py +++ b/module/remote/apitypes_debug.py @@ -60,7 +60,6 @@ methods = { 'checkOnlineStatus': OnlineCheck, 'checkOnlineStatusContainer': OnlineCheck, 'checkURLs': (dict, basestring, list), - 'configurePlugin': ConfigHolder, 'createPackage': int, 'deleteCollLink': None, 'deleteCollPack': None, @@ -99,6 +98,7 @@ methods = { 'getWSAddress': basestring, 'hasAddonHandler': bool, 'isInteractionWaiting': bool, + 'loadConfig': ConfigHolder, 'login': bool, 'moveFiles': bool, 'movePackage': bool, @@ -118,7 +118,6 @@ methods = { 'restartPackage': None, 'saveConfig': None, 'searchSuggestions': (list, basestring), - 'setConfigHandler': None, 'setConfigValue': None, 'setInteractionResult': None, 'setPackageFolder': bool, diff --git a/module/remote/pyload.thrift b/module/remote/pyload.thrift index 2aeb54091..adaede0ff 100644 --- a/module/remote/pyload.thrift +++ b/module/remote/pyload.thrift @@ -244,7 +244,7 @@ struct ConfigItem { } struct ConfigHolder { - 1: string name, + 1: string name, // for plugin this is the PluginName 2: string label, 3: string description, 4: string long_description, @@ -368,12 +368,11 @@ service Pyload { list<ConfigInfo> getPluginConfig(), list<ConfigInfo> getAvailablePlugins(), - ConfigHolder configurePlugin(1: PluginName plugin), + ConfigHolder loadConfig(1: string name), void setConfigValue(1: string section, 2: string option, 3: string value), void saveConfig(1: ConfigHolder config), void deleteConfig(1: PluginName plugin), - void setConfigHandler(1: PluginName plugin, 2: InteractionID iid, 3: JSONString value), /////////////////////// // Download Preparing diff --git a/module/web/static/js/models/ConfigHolder.js b/module/web/static/js/models/ConfigHolder.js new file mode 100644 index 000000000..8beb31fb8 --- /dev/null +++ b/module/web/static/js/models/ConfigHolder.js @@ -0,0 +1,44 @@ +define(['jquery', 'backbone', 'underscore', 'app', './ConfigItem'], + function($, Backbone, _, App, ConfigItem) { + + return Backbone.Model.extend({ + + defaults: { + name: "", + label: "", + description: "", + long_description: null, + // simple list but no collection + items: null, + info: null + }, + + // Model Constructor + initialize: function() { + + }, + + // Loads it from server by name + fetch: function(options) { + options = App.apiRequest('loadConfig/"' + this.get('name') + '"', null, options); + return Backbone.Model.prototype.fetch.call(this, options); + }, + + save: function(options) { + // TODO + }, + + parse: function(resp) { + // Create item models + resp.items = _.map(resp.items, function(item) { + return new ConfigItem(item); + }); + + return Backbone.Model.prototype.parse.call(this, resp); + }, + + isLoaded: function() { + return this.has('items') || this.has('long_description'); + } + }); + });
\ No newline at end of file diff --git a/module/web/static/js/models/ConfigItem.js b/module/web/static/js/models/ConfigItem.js new file mode 100644 index 000000000..f55bb2b9e --- /dev/null +++ b/module/web/static/js/models/ConfigItem.js @@ -0,0 +1,22 @@ +define(['jquery', 'backbone', 'underscore', 'app', 'utils/apitypes'], + function($, Backbone, _, App, Api) { + + return Backbone.Model.extend({ + + defaults: { + name: "", + label: "", + description: "", + input: null, + default_valie: null, + value: null, + // additional attributes + inputView: null + }, + + // Model Constructor + initialize: function() { + + } + }); + });
\ No newline at end of file diff --git a/module/web/static/js/views/headerView.js b/module/web/static/js/views/headerView.js index 25127a337..db704a3db 100644 --- a/module/web/static/js/views/headerView.js +++ b/module/web/static/js/views/headerView.js @@ -55,7 +55,8 @@ define(['jquery', 'underscore', 'backbone', 'app', 'models/ServerStatus', 'colle // TODO compare with polling ws.onmessage = _.bind(this.onData, this); ws.onerror = function(error) { - alert(error); + console.log(error); + alert("WebSocket error" + error); }; this.ws = ws; diff --git a/module/web/static/js/views/queryModal.js b/module/web/static/js/views/queryModal.js index 86fd5b78b..5477334a0 100644 --- a/module/web/static/js/views/queryModal.js +++ b/module/web/static/js/views/queryModal.js @@ -2,6 +2,7 @@ define(['jquery', 'underscore', 'app', 'views/abstract/modalView', './input/inpu function($, _, App, modalView, load_input, template) { return modalView.extend({ + // TODO: submit on enter reloads the page sometimes events: { 'click .btn-success': 'submit', 'submit form': 'submit' diff --git a/module/web/static/js/views/settingsView.js b/module/web/static/js/views/settingsView.js index a322cdae7..00c4b3739 100644 --- a/module/web/static/js/views/settingsView.js +++ b/module/web/static/js/views/settingsView.js @@ -1,23 +1,31 @@ -define(['jquery', 'underscore', 'backbone'], - function($, _, Backbone) { +define(['jquery', 'underscore', 'backbone', 'app', './input/inputLoader', 'models/ConfigHolder'], + function($, _, Backbone, App, load_input, ConfigHolder) { // Renders settings over view page return Backbone.View.extend({ el: "#content", - template_menu: _.compile($("#template-menu").html()), + templateMenu: _.compile($("#template-menu").html()), + templateConfig: _.compile($("#template-config").html()), + templateConfigItem: _.compile($("#template-config-item").html()), events: { 'click .settings-menu li > a': 'change_section' }, menu: null, + content: null, core_config: null, // It seems models are not needed plugin_config: null, + // currently open configHolder + config: null, + isLoading: false, + initialize: function() { - this.menu = $('.settings-menu'); + this.menu = this.$('.settings-menu'); + this.content = this.$('#settings-form'); this.refresh(); console.log("Settings initialized"); @@ -25,27 +33,80 @@ define(['jquery', 'underscore', 'backbone'], refresh: function() { var self = this; - $.ajax("/api/getCoreConfig", {success: function(data) { + $.ajax(App.apiRequest("getCoreConfig", null, {success: function(data) { self.core_config = data; - self.render() - }}); - $.ajax("/api/getPluginConfig", {success: function(data) { + self.render(); + }})); + $.ajax(App.apiRequest("getPluginConfig", null, {success: function(data) { self.plugin_config = data; self.render(); - }}); + }})); }, render: function() { - this.menu.html(this.template_menu({ - core: this.core_config, - plugin: this.plugin_config - })); + this.menu.html(this.templateMenu({ + core: this.core_config, + plugin: this.plugin_config + })); + }, + + openConfig: function(name) { + // Do nothing when this config is already open + if (this.config && this.config.get('name') === name) + return; + + this.config = new ConfigHolder({name: name}); + this.loading(); + + var self = this; + this.config.fetch({success: function() { + if (!self.isLoading) + self.show(); + + }, failure: _.bind(this.failure, this)}); + + }, + + loading: function() { + this.isLoading = true; + var self = this; + this.content.fadeOut({complete: function() { + if (self.config.isLoaded()) + self.show(); + + self.isLoading = false; + }}); + + }, + + show: function() { + // TODO: better refactor in separate views + this.content.html(this.templateConfig(this.config.toJSON())); + var container = this.content.find('.control-content'); + var items = this.config.get('items'); + var self = this; + _.each(items, function(item) { + var el = $('<div>').html(self.templateConfigItem(item.toJSON())); + var inputView = load_input("todo"); + el.find('.controls').append( + new inputView(item.get('input'), item.get('value'), + item.get('default_value'), item.get('description')).render().el); + container.append(el); + }); + + this.content.fadeIn(); + }, + + failure: function() { + }, change_section: function(e) { + // TODO check for changes + var el = $(e.target).parent(); var name = el.data("name"); - console.log("Section changed to " + name); + this.openConfig(name); this.menu.find("li.active").removeClass("active"); el.addClass("active"); diff --git a/module/web/templates/default/settings.html b/module/web/templates/default/settings.html index 1f9be3db0..7f6bbeb8e 100644 --- a/module/web/templates/default/settings.html +++ b/module/web/templates/default/settings.html @@ -15,119 +15,59 @@ {% block head %}
<script type="text/template" id="template-menu">
<%=if core%>
- <li class="nav-header"><i class="icon-globe icon-white"></i> {{ _("General") }}</li>
- <%=each core%>
- <li data-name="<% this.name %>"><a href="#"><% this.label %></a></li>
- <%/each%>
+ <li class="nav-header"><i class="icon-globe icon-white"></i> {{ _("General") }}</li>
+ <%=each core%>
+ <li data-name="<% this.name %>"><a href="#"><% this.label %></a></li>
+ <%/each%>
<%/if%>
<li class="divider"></li>
<li class="nav-header"><i class="icon-th-large icon-white"></i> {{ _("Addons") }}</li>
<li class="divider"></li>
<li class="nav-header"><i class="icon-th-list icon-white"></i> {{ _("Other") }}</li>
</script>
-{% endblock %}
-
-{% block content %}
- <div class="span2">
- <ul class="nav nav-list well settings-menu">
- </ul>
+ <script type="text/template" id="template-config">
+ <legend>
+ <div class="page-header">
+ <h1><% label %>
+ <small><% description %></small>
+ <a class="btn btn-small " href="#"><i
+ class="icon-question-sign"></i></a>
+ </h1>
</div>
- <!-- Info Popup -->
- <div class="modal hide fade in" id="info">
- <div class="modal-header">
- <button type="button" class="close">×</button>
- <h3>Info Popup</h3>
- </div>
- <div class="modal-body">
- <h4>General</h4>
-
- <p>Duis mollis, est non commodo luctus, nisi erat porttitor ligula, eget lacinia odio sem.</p>
-
- <h4>And...</h4>
-
- <p>Cras mattis consectetur purus sit amet fermentum. Cras justo odio, dapibus ac facilisis in,
- egestas eget quam. Morbi leo risus, porta ac consectetur ac, vestibulum at eros.</p>
-
- </div>
- <div class="modal-footer">
- <button class="btn">Close</button>
- </div>
- </div>
- <!-- End Info Popup -->
- <div class="span10">
- <div class="well setting-box">
- <form class="form-horizontal">
- <legend>
- <div class="page-header">
- <h1>Example Settings
- <small>Subtext for header</small>
- <a class="btn btn-small " href="#"><i
- class="icon-question-sign"></i></a>
- </h1>
- </div>
- </legend>
- <div class="control-group">
- <label class="control-label">Max Parallel Downloads</label>
-
- <div class="controls">
- <input id="in_mpd" type="text" placeholder="3">
- </div>
- </div>
- <div class="control-group">
- <label class="control-label">Limit Download Speed</label>
-
- <div class="controls">
- <div class="btn-group" data-toggle="buttons-radio">
- <button type="button" class="btn bnmaxspeed" id="onmaxspeed">On</button>
- <button type="button" class="btn bnmaxspeed active" id="offmaxspeed">Off</button>
- </div>
- <div id="downloadspeed" style="display:none">
- <label>Max Download Speed in kb/s</label>
- <input type="text" placeholder="Tipp etwas ...">
- </div>
- </div>
- </div>
- <div class="control-group">
- <label class="control-label">Allow IPv6</label>
-
- <div class="controls">
- <div class="btn-group" data-toggle="buttons-radio">
- <button type="button" class="btn bnip6" id="onip6">On</button>
- <button type="button" class="btn bnip6 active" id="offip6">Off</button>
- </div>
- </div>
- </div>
- <div class="form-actions">
- <button type="submit" class="btn btn-primary">Änderungen Speichern</button>
- <button type="button" class="btn">Abbrechen</button>
- </div>
- </form>
- </div>
+ </legend>
+ <div class="control-content">
+ </div>
+ <div class="form-actions">
+ <button type="submit" class="btn btn-primary disabled">Save changes</button>
+ <button type="button" class="btn">Cancel</button>
+ </div>
+ </script>
+ <script type="text/template" id="template-config-item">
+ <div class="control-group">
+ <label class="control-label"><% label %></label>
+ <div class="controls">
</div>
- <script src="static/js/libs/jquery-1.9.0.js"></script>
- {# <script src="static/js/libs/bootstrap-2.1.1.js"></script>#}
- <script type="text/javascript">
- $(".bnmaxspeed").click(function() {
- $(".bnmaxspeed").removeClass("active");
- $(this).toggleClass("active");
- if ($("#onmaxspeed").hasClass("active")) {
- $("#downloadspeed").show();
- }
- else {
- $("#downloadspeed").hide();
- }
+ </div>
+ </script>
- });
- $(".bnip6").click(function() {
- $(".bnip6").removeClass("active");
- $(this).toggleClass("active");
- });
- $('#info').modal('toggle');
+{% endblock %}
- $('#in_mpd').tooltip({
- placement: 'right',
- title: 'Gib an wie viele Downloads gleichzeitg laufen dürfen.'
- });
+{% block actionbar %}
+{# <ul class="actionbar nav nav-pills span9">#}
+{# <li>Add Plugin</li>#}
+{# </ul>#}
+{% endblock %}
- </script>
+{% block content %}
+ <div class="span2">
+ <ul class="nav nav-list well settings-menu">
+ </ul>
+ </div>
+ <div class="span10">
+ <div class="well setting-box">
+ <form class="form-horizontal" id="settings-form">
+ <h1>Please choose a config section</h1>
+ </form>
+ </div>
+ </div>
{% endblock %}
\ No newline at end of file diff --git a/tests/manager/test_configManager.py b/tests/manager/test_configManager.py index 0fed702c1..6c10da4dd 100644 --- a/tests/manager/test_configManager.py +++ b/tests/manager/test_configManager.py @@ -89,7 +89,6 @@ class TestConfigManager(TestCase): # should not trigger something self.config.delete("foo") - def test_sections(self): self.addConfig() @@ -111,6 +110,9 @@ class TestConfigManager(TestCase): i +=1 assert i == 1 + def test_get_section(self): + self.addConfig() + assert self.config.getSection("plugin")[0].name == "Name" @raises(InvalidConfigSection) def test_restricted_access(self): diff --git a/tests/other/test_configparser.py b/tests/other/test_configparser.py index 09b686738..7f34e64d3 100644 --- a/tests/other/test_configparser.py +++ b/tests/other/test_configparser.py @@ -31,6 +31,9 @@ class TestConfigParser(): assert isinstance(config.config, dict) assert isinstance(values, dict) + def test_get(self): + assert self.config.getSection("general")[0].config + @raises(KeyError) def test_invalid_config(self): print self.config["invalid"]["config"] |