Mercurial > libervia-backend
view src/tools/xml_tools.py @ 787:dd656d745d6a
test: added tests for XEP-0033
author | souliane <souliane@mailoo.org> |
---|---|
date | Sun, 05 Jan 2014 13:05:31 +0100 |
parents | bfabeedbf32e |
children | 46aa5ada61bf |
line wrap: on
line source
#!/usr/bin/python # -*- coding: utf-8 -*- # SAT: a jabber client # Copyright (C) 2009, 2010, 2011, 2012, 2013 Jérôme Poisson (goffi@goffi.org) # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. from sat.core.i18n import _ from logging import debug, info, error, warning from xml.dom import minidom from wokkel import data_form from twisted.words.xish import domish from sat.core import exceptions """This library help manage XML used in SàT (parameters, registration, etc) """ SAT_FORM_PREFIX ="SAT_FORM_" def dataForm2XMLUI(form, submit_id, session_id=None): """Take a data form (xep-0004, Wokkel's implementation) and convert it to a SàT xml""" form_ui = XMLUI("form", "vertical", submit_id=submit_id, session_id=session_id) if form.instructions: form_ui.addText('\n'.join(form.instructions), 'instructions') labels = [field for field in form.fieldList if field.label] if labels: # if there is no label, we don't need to use pairs form_ui.changeLayout("pairs") for field in form.fieldList: if field.fieldType == 'fixed': __field_type = 'text' elif field.fieldType == 'text-single': __field_type = "string" elif field.fieldType == 'text-private': __field_type = "password" elif field.fieldType == 'list-single': __field_type = "list" else: error(u"FIXME FIXME FIXME: Type [%s] is not managed yet by SàT" % field.fieldType) __field_type = "string" if labels: if field.label: form_ui.addLabel(field.label) else: form_ui.addEmpty() form_ui.addElement(__field_type, field.var, field.value, [option.value for option in field.options]) return form_ui def dataFormResult2AdvancedList(form_ui, form_xml): """Take a raw data form (not parsed by XEP-0004) and convert it to an advanced list raw data form is used because Wokkel doesn't manage result items parsing yet @param form_ui: the XMLUI where the AdvancedList will be added @param form_xml: domish.Element of the data form @return: AdvancedList element """ headers = [] items = [] try: reported_elt = form_xml.elements('jabber:x:data', 'reported').next() except StopIteration: raise exceptions.DataError("Couldn't find expected <reported> tag") for elt in reported_elt.elements(): if elt.name != "field": raise exceptions.DataError("Unexpected tag") name = elt["var"] label = elt.attributes.get('label','') type_ = elt.attributes.get('type','') # TODO headers.append(Header(name, label)) if not headers: raise exceptions.DataError("No reported fields (see XEP-0004 §3.4)") item_elts = form_xml.elements('jabber:x:data', 'item') for item_elt in item_elts: fields = [] for elt in item_elt.elements(): if elt.name != 'field': warning("Unexpected tag (%s)" % elt.name) continue name = elt['var'] child_elt = elt.firstChildElement() if child_elt.name != "value": raise exceptions.DataError('Was expecting <value> tag') value = unicode(child_elt) fields.append(Field(name, value)) items.append(Item(' | '.join((field.value for field in fields if field)), fields)) return form_ui.addAdvancedList(None, headers, items) def dataFormResult2XMLUI(form_xml, session_id=None): """Take a raw data form (not parsed by XEP-0004) and convert it to a SàT XMLUI raw data form is used because Wokkel doesn't manage result items parsing yet @param form_xml: domish.Element of the data form @return: XMLUI interface """ form_ui = XMLUI("window", "vertical", session_id=session_id) dataFormResult2AdvancedList(form_ui, form_xml) return form_ui def XMLUIResult2DataFormResult(xmlui_data): """ Extract form data from a XMLUI return @xmlui_data: data returned by frontends for XMLUI form @return: dict of data usable by Wokkel's dataform """ return {key[len(SAT_FORM_PREFIX):]: value for key, value in xmlui_data.iteritems() if key.startswith(SAT_FORM_PREFIX)} def XMLUIResultToElt(xmlui_data): """ Construct result domish.Element from XMLUI result @xmlui_data: data returned by frontends for XMLUI form """ form = data_form.Form('result') form.makeFields(XMLUIResult2DataFormResult(xmlui_data)) return form.toElement() def tupleList2dataForm(values): """convert a list of tuples (name,value) to a wokkel submit data form""" form = data_form.Form('submit') for value in values: field = data_form.Field(var=value[0], value=value[1]) form.addField(field) return form def paramsXml2xmlUI(xml): """Convert the xml for parameter to a SàT XML User Interface""" params_doc = minidom.parseString(xml.encode('utf-8')) top = params_doc.documentElement if top.nodeName != 'params': error(_('INTERNAL ERROR: parameters xml not valid')) assert(False) param_ui = XMLUI("param", "tabs") for category in top.getElementsByTagName("category"): name = category.getAttribute('name') label = category.getAttribute('label') if not name: error(_('INTERNAL ERROR: params categories must have a name')) assert(False) param_ui.addCategory(name, 'pairs', label=label) for param in category.getElementsByTagName("param"): name = param.getAttribute('name') label = param.getAttribute('label') if not name: error(_('INTERNAL ERROR: params must have a name')) assert(False) type_ = param.getAttribute('type') value = param.getAttribute('value') or None options = getOptions(param) callback_id = param.getAttribute('callback_id') or None if type_ == "button": param_ui.addEmpty() else: param_ui.addLabel(label or name) param_ui.addElement(name=name, type_=type_, value=value, options=options, callback_id=callback_id) return param_ui.toXml() def getOptions(param): """Retrieve the options for list element. Allow listing the <option/> tags directly in <param/> or in an intermediate <options/> tag.""" elems = param.getElementsByTagName("options") if len(elems) == 0: elems = param.getElementsByTagName("option") else: elems = elems.item(0).getElementsByTagName("option") if len(elems) == 0: return [] return [elem.getAttribute("value") for elem in elems] class Header(object): """AdvandeList's header""" def __init__(self, field_name, label, description=None, type_=None): """ @param field_name: name of the field referenced @param label: label to be displayed in columns @param description: long descriptive text @param type_: TODO """ self.field_name = field_name self.label = label self.description = description self.type = type_ if type_ is not None: raise NotImplementedError # TODO: class Item(object): """Item used in AdvancedList""" def __init__(self, text=None, fields=None): """ @param text: Optional textual representation, when fields are not showed individually @param fields: list of Field instances² """ self.text = text self.fields = fields if fields is not None else [] class Field(object): """Field used in AdvancedList (in items)""" def __init__(self, name, value): """ @param name: name of the field, used to identify the field in headers @param value: actual content of the field """ self.name = name self.value = value class XMLUI(object): """This class is used to create a user interface (form/window/parameters/etc) using SàT XML""" def __init__(self, panel_type, layout="vertical", title=None, submit_id=None, session_id=None): """Init SàT XML Panel @param panel_type: one of - window (new window) - form (formulaire, depend of the frontend, usually a panel with cancel/submit buttons) - param (parameters, presentatio depend of the frontend) @param layout: disposition of elements, one of: - vertical: elements are disposed up to bottom - horizontal: elements are disposed left to right - pairs: elements come on two aligned columns (usually one for a label, the next for the element) - tabs: elemens are in categories with tabs (notebook) @param title: title or default if None @param submit_id: callback id to call for panel_type we can submit (form, param) """ if panel_type not in ['window', 'form', 'param']: raise exceptions.DataError(_("Unknown panel type [%s]") % panel_type) if panel_type == 'form' and submit_id is None: raise exceptions.DataError(_("form XMLUI need a submit_id")) self.type_ = panel_type impl = minidom.getDOMImplementation() self.doc = impl.createDocument(None, "sat_xmlui", None) top_element = self.doc.documentElement top_element.setAttribute("type", panel_type) if title: top_element.setAttribute("title", title) if submit_id: top_element.setAttribute("submit", submit_id) if session_id is not None: top_element.setAttribute("session_id", session_id) self.parentTabsLayout = None # used only we have 'tabs' layout self.currentCategory = None # used only we have 'tabs' layout self.currentLayout = None self.changeLayout(layout) def __del__(self): self.doc.unlink() def setSessionId(self, session_id): assert(session_id) top_element = self.doc.documentElement top_element.setAttribute("session_id", session_id) def _createLayout(self, layout, parent=None): """Create a layout element @param type: layout type (cf init doc) @parent: parent element or None """ if not layout in ['vertical', 'horizontal', 'pairs', 'tabs']: error(_("Unknown layout type [%s]") % layout) assert False layout_elt = self.doc.createElement('layout') layout_elt.setAttribute('type', layout) if parent is not None: parent.appendChild(layout_elt) return layout_elt def _createElem(self, type_, name=None, parent=None): """Create an element @param type_: one of - empty: empty element (usefull to skip something in a layout, e.g. skip first element in a PAIRS layout) - text: text to be displayed in an multi-line area, e.g. instructions @param name: name of the element or None @param parent: parent element or None @return: created element """ elem = self.doc.createElement('elem') if name: elem.setAttribute('name', name) elem.setAttribute('type', type_) if parent is not None: parent.appendChild(elem) return elem def changeLayout(self, layout): """Change the current layout""" self.currentLayout = self._createLayout(layout, self.currentCategory if self.currentCategory else self.doc.documentElement) if layout == "tabs": self.parentTabsLayout = self.currentLayout def addEmpty(self, name=None): """Add a multi-lines text""" return self._createElem('empty', name, self.currentLayout) def addText(self, text, name=None): """Add a multi-lines text""" elem = self._createElem('text', name, self.currentLayout) text = self.doc.createTextNode(text) elem.appendChild(text) return elem def addLabel(self, text, name=None): """Add a single line text, mainly useful as label before element""" elem = self._createElem('label', name, self.currentLayout) elem.setAttribute('value', text) return elem def addString(self, name=None, value=None): """Add a string box""" elem = self._createElem('string', name, self.currentLayout) if value: elem.setAttribute('value', value) return elem def addPassword(self, name=None, value=None): """Add a password box""" elem = self._createElem('password', name, self.currentLayout) if value: elem.setAttribute('value', value) return elem def addTextBox(self, name=None, value=None): """Add a string box""" elem = self._createElem('textbox', name, self.currentLayout) if value: elem.setAttribute('value', value) return elem def addBool(self, name=None, value="true"): """Add a string box""" assert value in ["true", "false"] elem = self._createElem('bool', name, self.currentLayout) elem.setAttribute('value', value) return elem def addList(self, options, name=None, value=None, style=None): """Add a list of choices""" if style is None: style = set() styles = set(style) assert options assert styles.issubset(['multi']) elem = self._createElem('list', name, self.currentLayout) self.addOptions(options, elem) if value: elem.setAttribute('value', value) for style in styles: elem.setAttribute(style, 'yes') return elem def addAdvancedList(self, name=None, headers=None, items=None): """Create an advanced list @param headers: optional headers informations @param items: list of Item instances @return: created element """ elem = self._createElem('advanced_list', name, self.currentLayout) self.addHeaders(headers, elem) if items: self.addItems(items, elem) return elem def addButton(self, callback_id, name, value, fields_back=[]): """Add a button @param callback: callback which will be called if button is pressed @param name: name @param value: label of the button @fields_back: list of names of field to give back when pushing the button """ elem = self._createElem('button', name, self.currentLayout) elem.setAttribute('callback', callback_id) elem.setAttribute('value', value) for field in fields_back: fback_el = self.doc.createElement('field_back') fback_el.setAttribute('name', field) elem.appendChild(fback_el) return elem def addElement(self, type_, name=None, value=None, options=None, callback_id=None, headers=None, available=None): """Convenience method to add element, the params correspond to the ones in addSomething methods""" if type_ == 'empty': return self.addEmpty(name) elif type_ == 'text': assert value is not None return self.addText(value, name) elif type_ == 'label': assert(value) return self.addLabel(value) elif type_ == 'string': return self.addString(name, value) elif type_ == 'password': return self.addPassword(name, value) elif type_ == 'textbox': return self.addTextBox(name, value) elif type_ == 'bool': if not value: value = "true" return self.addBool(name, value) elif type_ == 'list': return self.addList(options, name, value) elif type_ == 'advancedlist': return self.addAdvancedList(name, headers, available) elif type_ == 'button': assert(callback_id and value) return self.addButton(callback_id, name, value) # List def addOptions(self, options, parent): """Add options to a multi-values element (e.g. list) @param parent: multi-values element""" for option in options: opt = self.doc.createElement('option') if isinstance(option, basestring): value, label = option, option elif isinstance(option, tuple): value, label = option opt.setAttribute('value', value) opt.setAttribute('label', label) parent.appendChild(opt) # Advanced list def addHeaders(self, headers, parent): headers_elt = self.doc.createElement('headers') for header in headers: field_elt = self.doc.createElement('field') field_elt.setAttribute('field_name', header.field_name) field_elt.setAttribute('label', header.label) if header.description: field_elt.setAttribute('description', header.description) if header.type: field_elt.setAttribute('type', header.type) headers_elt.appendChild(field_elt) parent.appendChild(headers_elt) def addItems(self, items, parent): """Add items to an AdvancedList @param items: list of Item instances @param parent: parent element (should be addAdvancedList) """ items_elt = self.doc.createElement('items') for item in items: item_elt = self.doc.createElement('item') if item.text is not None: text_elt = self.doc.createElement('text') text_elt.appendChild(self.doc.createTextNode(item.text)) item_elt.appendChild(text_elt) for field in item.fields: field_elt = self.doc.createElement('field') field_elt.setAttribute('name', field.name) field_elt.setAttribute('value', field.value) item_elt.appendChild(field_elt) items_elt.appendChild(item_elt) parent.appendChild(items_elt) # Tabs def addCategory(self, name, layout, label=None): """Add a category to current layout (must be a tabs layout)""" assert(layout != 'tabs') if not self.parentTabsLayout: error(_("Trying to add a category without parent tabs layout")) assert(False) if self.parentTabsLayout.getAttribute('type') != 'tabs': error(_("parent layout of a category is not tabs")) assert(False) if not label: label = name self.currentCategory = cat = self.doc.createElement('category') cat.setAttribute('name', name) cat.setAttribute('label', label) self.changeLayout(layout) self.parentTabsLayout.appendChild(cat) def toXml(self): """return the XML representation of the panel""" return self.doc.toxml() class ElementParser(object): """callable class to parse XML string into Element Found at http://stackoverflow.com/questions/2093400/how-to-create-twisted-words-xish-domish-element-entirely-from-raw-xml/2095942#2095942 (c) Karl Anderson""" def __call__(self, string): self.result = None def onStart(elem): self.result = elem def onEnd(): pass def onElement(elem): self.result.addChild(elem) parser = domish.elementStream() parser.DocumentStartEvent = onStart parser.ElementEvent = onElement parser.DocumentEndEvent = onEnd tmp = domish.Element((None, "s")) tmp.addRawXml(string.replace('\n', ' ').replace('\t', ' ')) parser.parse(tmp.toXml().encode('utf-8')) return self.result.firstChildElement()