SSSD info plugin is immutable if not preconfigured
[cascardo/ipsilon.git] / ipsilon / admin / common.py
1 # Copyright (C) 2014  Simo Sorce <simo@redhat.com>
2 #
3 # see file 'COPYING' for use and warranty information
4 #
5 # This program is free software; you can redistribute it and/or modify
6 # it under the terms of the GNU General Public License as published by
7 # the Free Software Foundation, either version 3 of the License, or
8 # (at your option) any later version.
9 #
10 # This program is distributed in the hope that it will be useful,
11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 # GNU General Public License for more details.
14 #
15 # You should have received a copy of the GNU General Public License
16 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
17
18 import cherrypy
19 from ipsilon.util.page import Page
20 from ipsilon.util.page import admin_protect
21 from ipsilon.util.endpoint import allow_iframe
22 from ipsilon.util import config as pconfig
23
24
25 ADMIN_STATUS_OK = "success"
26 ADMIN_STATUS_ERROR = "danger"
27 ADMIN_STATUS_WARN = "warning"
28
29
30 class AdminError(Exception):
31     def __init__(self, message):
32         super(AdminError, self).__init__(message)
33         self.message = message
34
35     def __str__(self):
36         return str(self.message)
37
38
39 class AdminPage(Page):
40
41     def __init__(self, *args, **kwargs):
42         super(AdminPage, self).__init__(*args, **kwargs)
43         self.auth_protect = True
44
45
46 class AdminPluginConfig(AdminPage):
47
48     def __init__(self, po, site, parent):
49         super(AdminPluginConfig, self).__init__(site, form=True)
50         self._po = po
51         self.title = '%s plugin' % po.name
52         self.url = '%s/%s' % (parent.url, po.name)
53         self.facility = parent.facility
54         self.menu = [parent]
55         self.back = parent.url
56
57     def root_with_msg(self, message=None, message_type=None):
58         return self._template('admin/option_config.html', title=self.title,
59                               menu=self.menu, action=self.url, back=self.back,
60                               message=message, message_type=message_type,
61                               name='admin_%s_%s_form' % (self.facility,
62                                                          self._po.name),
63                               config=self._po.get_config_obj())
64
65     @admin_protect
66     def GET(self, *args, **kwargs):
67         return self.root_with_msg()
68
69     @admin_protect
70     def POST(self, *args, **kwargs):
71
72         if self._po.is_readonly:
73             return self.root_with_msg(
74                 message="Configuration is marked Read-Only",
75                 message_type=ADMIN_STATUS_WARN)
76
77         message = "Nothing was modified."
78         message_type = "info"
79         new_db_values = dict()
80
81         conf = self._po.get_config_obj()
82
83         for name, option in conf.iteritems():
84             if name in kwargs:
85                 value = kwargs[name]
86                 if isinstance(option, pconfig.List):
87                     value = [x.strip() for x in value.split('\n')]
88                 elif isinstance(option, pconfig.Condition):
89                     value = True
90             else:
91                 if isinstance(option, pconfig.Condition):
92                     value = False
93                 elif isinstance(option, pconfig.Choice):
94                     value = list()
95                     for a in option.get_allowed():
96                         aname = '%s_%s' % (name, a)
97                         if aname in kwargs:
98                             value.append(a)
99                 elif type(option) is pconfig.ComplexList:
100                     value = get_complex_list_value(name,
101                                                    option.get_value(),
102                                                    **kwargs)
103                     if value is None:
104                         continue
105                 elif type(option) is pconfig.MappingList:
106                     value = get_mapping_list_value(name,
107                                                    option.get_value(),
108                                                    **kwargs)
109                     if value is None:
110                         continue
111                 else:
112                     continue
113
114             if value != option.get_value():
115                 cherrypy.log.error("Storing [%s]: %s = %s" %
116                                    (self._po.name, name, value))
117             option.set_value(value)
118             new_db_values[name] = option.export_value()
119
120         if len(new_db_values) != 0:
121             # First we try to save in the database
122             try:
123                 self._po.save_plugin_config(new_db_values)
124                 message = "New configuration saved."
125                 message_type = ADMIN_STATUS_OK
126             except Exception as e:  # pylint: disable=broad-except
127                 self.error('Failed to save data: %s' % e)
128                 message = "Failed to save data!"
129                 message_type = ADMIN_STATUS_ERROR
130
131             # Then refresh the actual objects
132             self._po.refresh_plugin_config()
133
134         return self.root_with_msg(message=message,
135                                   message_type=message_type)
136
137
138 class AdminPluginsOrder(AdminPage):
139
140     def __init__(self, site, parent, facility):
141         super(AdminPluginsOrder, self).__init__(site, form=True)
142         self.parent = parent
143         self.facility = facility
144         self.url = '%s/order' % parent.url
145         self.menu = [parent]
146
147     @admin_protect
148     def GET(self, *args, **kwargs):
149         return self.parent.root_with_msg()
150
151     @admin_protect
152     def POST(self, *args, **kwargs):
153
154         if self._site[self.facility].is_readonly:
155             return self.parent.root_with_msg(
156                 message="Configuration is marked Read-Only",
157                 message_type=ADMIN_STATUS_WARN)
158
159         message = "Nothing was modified."
160         message_type = "info"
161         changed = None
162         cur_enabled = self._site[self.facility].enabled
163
164         if 'order' in kwargs:
165             order = kwargs['order'].split(',')
166             if len(order) != 0:
167                 new_order = []
168                 try:
169                     for v in order:
170                         val = v.strip()
171                         if val not in cur_enabled:
172                             error = "Invalid plugin name: %s" % val
173                             raise ValueError(error)
174                         new_order.append(val)
175                     if len(new_order) < len(cur_enabled):
176                         for val in cur_enabled:
177                             if val not in new_order:
178                                 new_order.append(val)
179
180                     self.parent.save_enabled_plugins(new_order)
181
182                     # When all is saved update also live config. The
183                     # live config is the ordered list of plugin names.
184                     self._site[self.facility].refresh_enabled()
185
186                     message = "New configuration saved."
187                     message_type = ADMIN_STATUS_OK
188
189                     changed = dict()
190                     self.debug('%s -> %s' % (cur_enabled, new_order))
191                     for i in range(0, len(cur_enabled)):
192                         if cur_enabled[i] != new_order[i]:
193                             changed[cur_enabled[i]] = 'reordered'
194
195                 except ValueError, e:
196                     message = str(e)
197                     message_type = ADMIN_STATUS_ERROR
198
199                 except Exception as e:  # pylint: disable=broad-except
200                     self.error('Failed to save data: %s' % e)
201                     message = "Failed to save data!"
202                     message_type = ADMIN_STATUS_ERROR
203
204         return self.parent.root_with_msg(message=message,
205                                          message_type=message_type,
206                                          changed=changed)
207
208
209 class AdminPlugins(AdminPage):
210     def __init__(self, name, site, parent, facility, ordered=True):
211         super(AdminPlugins, self).__init__(site)
212         self._master = parent
213         self.name = name
214         self.title = '%s plugins' % name
215         self.url = '%s/%s' % (parent.url, name)
216         self.facility = facility
217         self.template = 'admin/plugins.html'
218         self.order = None
219         parent.add_subtree(name, self)
220
221         if self._site[facility] is None:
222             return
223
224         for plugin in self._site[facility].available:
225             cherrypy.log.error('Admin info plugin: %s' % plugin)
226             obj = self._site[facility].available[plugin]
227             page = AdminPluginConfig(obj, self._site, self)
228             if hasattr(obj, 'admin'):
229                 obj.admin.mount(page)
230             self.add_subtree(plugin, page)
231
232         if ordered:
233             self.order = AdminPluginsOrder(self._site, self, facility)
234
235     def save_enabled_plugins(self, names):
236         self._site[self.facility].save_enabled(names)
237
238     def root_with_msg(self, message=None, message_type=None, changed=None):
239         plugins = self._site[self.facility]
240
241         if changed is None:
242             changed = dict()
243
244         targs = {'title': self.title,
245                  'menu': self._master.menu,
246                  'message': message,
247                  'message_type': message_type,
248                  'available': plugins.available,
249                  'enabled': plugins.enabled,
250                  'changed': changed,
251                  'baseurl': self.url,
252                  'newurl': self.url}
253         if self.order:
254             targs['order_name'] = '%s_order_form' % self.name
255             targs['order_action'] = self.order.url
256
257         # pylint: disable=star-args
258         return self._template(self.template, **targs)
259
260     def root(self, *args, **kwargs):
261         return self.root_with_msg()
262
263     def _get_plugin_obj(self, plugin):
264         plugins = self._site[self.facility]
265         if plugins.is_readonly:
266             msg = "Configuration is marked Read-Only"
267             raise AdminError(msg)
268         if plugin not in plugins.available:
269             msg = "Unknown plugin %s" % plugin
270             raise AdminError(msg)
271         obj = plugins.available[plugin]
272         if obj.is_readonly:
273             msg = "Plugin Configuration is marked Read-Only"
274             raise AdminError(msg)
275         return obj
276
277     @admin_protect
278     def enable(self, plugin):
279         msg = None
280         try:
281             obj = self._get_plugin_obj(plugin)
282         except AdminError, e:
283             return self.root_with_msg(str(e), ADMIN_STATUS_WARN)
284         if not obj.is_enabled:
285             try:
286                 obj.enable()
287             except Exception as e:  # pylint: disable=broad-except
288                 return self.root_with_msg(str(e), ADMIN_STATUS_WARN)
289             obj.save_enabled_state()
290             msg = "Plugin %s enabled" % obj.name
291         return self.root_with_msg(msg, ADMIN_STATUS_OK,
292                                   changed={obj.name: 'enabled'})
293     enable.public_function = True
294
295     @admin_protect
296     def disable(self, plugin):
297         msg = None
298         try:
299             obj = self._get_plugin_obj(plugin)
300         except AdminError, e:
301             return self.root_with_msg(str(e), ADMIN_STATUS_WARN)
302         if obj.is_enabled:
303             try:
304                 obj.disable()
305             except Exception as e:  # pylint: disable=broad-except
306                 return self.root_with_msg(str(e), ADMIN_STATUS_WARN)
307             obj.save_enabled_state()
308             msg = "Plugin %s disabled" % obj.name
309         return self.root_with_msg(msg, ADMIN_STATUS_OK,
310                                   changed={obj.name: 'disabled'})
311     disable.public_function = True
312
313
314 class Admin(AdminPage):
315
316     def __init__(self, site, mount):
317         super(Admin, self).__init__(site)
318         self.title = 'Home'
319         self.mount = mount
320         self.url = '%s/%s' % (self.basepath, mount)
321         self.menu = [self]
322
323     def root(self, *args, **kwargs):
324         return self._template('admin/index.html',
325                               title='Configuration',
326                               baseurl=self.url,
327                               menu=self.menu)
328
329     def add_subtree(self, name, page):
330         self.__dict__[name] = page
331         self.menu.append(page)
332
333     def del_subtree(self, name):
334         self.menu.remove(self.__dict__[name])
335         del self.__dict__[name]
336
337     def get_menu_urls(self):
338         urls = dict()
339         for item in self.menu:
340             name = getattr(item, 'name', None)
341             if name:
342                 urls['%s_url' % name] = cherrypy.url('/%s/%s' % (self.mount,
343                                                                  name))
344         return urls
345
346     @admin_protect
347     @allow_iframe
348     def scheme(self):
349         cherrypy.response.headers.update({'Content-Type': 'image/svg+xml'})
350         urls = self.get_menu_urls()
351         # pylint: disable=star-args
352         return str(self._template('admin/ipsilon-scheme.svg', **urls))
353     scheme.public_function = True
354
355
356 def get_complex_list_value(name, old_value, **kwargs):
357     delete = list()
358     change = dict()
359     for key, val in kwargs.iteritems():
360         if not key.startswith(name):
361             continue
362         n = key[len(name):]
363         if len(n) == 0 or n[0] != ' ':
364             continue
365         try:
366             index, field = n[1:].split('-')
367         except ValueError:
368             continue
369         if field == 'delete':
370             delete.append(int(index))
371         elif field == 'name':
372             change[int(index)] = val
373
374     if len(delete) == 0 and len(change) == 0:
375         return None
376
377     value = old_value
378
379     # remove unwanted changes
380     for i in delete:
381         if i in change:
382             del change[i]
383
384     # perform requested changes
385     for index, val in change.iteritems():
386         val_list = val.split('/')
387         stripped = list()
388         for v in val_list:
389             stripped.append(v.strip())
390         if len(stripped) == 1:
391             stripped = stripped[0]
392         if len(value) <= index:
393             value.extend([None]*(index + 1 - len(value)))
394         value[index] = stripped
395
396         if len(value[index]) == 0:
397             value[index] = None
398
399     # the previous loop may add 'None' entries
400     # if any still exists mark them to be deleted
401     for i in xrange(0, len(value)):
402         if value[i] is None:
403             delete.append(i)
404
405     # remove duplicates and set in reverse order
406     delete = list(set(delete))
407     delete.sort(reverse=True)
408
409     for i in delete:
410         if len(value) > i:
411             del value[i]
412
413     if len(value) == 0:
414         value = None
415     return value
416
417
418 def get_mapping_list_value(name, old_value, **kwargs):
419     delete = list()
420     change = dict()
421     for key, val in kwargs.iteritems():
422         if not key.startswith(name):
423             continue
424         n = key[len(name):]
425         if len(n) == 0 or n[0] != ' ':
426             continue
427         try:
428             index, field = n[1:].split('-')
429         except ValueError:
430             continue
431         if field == 'delete':
432             delete.append(int(index))
433         else:
434             i = int(index)
435             if i not in change:
436                 change[i] = dict()
437             change[i][field] = val
438
439     if len(delete) == 0 and len(change) == 0:
440         return None
441
442     value = old_value
443
444     # remove unwanted changes
445     for i in delete:
446         if i in change:
447             del change[i]
448
449     # perform requested changes
450     for index, fields in change.iteritems():
451         for k in 'from', 'to':
452             if k in fields:
453                 val = fields[k]
454                 val_list = val.split('/')
455                 stripped = list()
456                 for v in val_list:
457                     stripped.append(v.strip())
458                 if len(stripped) == 1:
459                     stripped = stripped[0]
460                 if len(value) <= index:
461                     value.extend([None]*(index + 1 - len(value)))
462                 if value[index] is None:
463                     value[index] = [None, None]
464                 if k == 'from':
465                     i = 0
466                 else:
467                     i = 1
468                 value[index][i] = stripped
469
470         # eliminate incomplete/incorrect entries
471         if value[index] is not None:
472             if ((len(value[index]) != 2 or
473                  value[index][0] is None or
474                  len(value[index][0]) == 0 or
475                  value[index][1] is None or
476                  len(value[index][1]) == 0)):
477                 value[index] = None
478
479     # the previous loop may add 'None' entries
480     # if any still exists mark them to be deleted
481     for i in xrange(0, len(value)):
482         if value[i] is None:
483             delete.append(i)
484
485     # remove duplicates and set in reverse order
486     delete = list(set(delete))
487     delete.sort(reverse=True)
488
489     for i in delete:
490         if len(value) > i:
491             del value[i]
492
493     if len(value) == 0:
494         value = None
495     return value