2020-04-28 08:46:30 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# Copyright 2019-2020 Fabien Bourgeois <fabien@yaltik.com>
|
|
|
|
#
|
|
|
|
# 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/>.
|
|
|
|
|
2020-05-03 12:03:47 +02:00
|
|
|
""" XML helpers and macros """
|
2020-04-28 08:46:30 +02:00
|
|
|
|
|
|
|
from os import path
|
|
|
|
import xml.etree.ElementTree as ET
|
|
|
|
from xml.dom import minidom
|
2020-05-03 12:03:47 +02:00
|
|
|
from collections import namedtuple
|
|
|
|
from functools import partial
|
2020-04-28 08:46:30 +02:00
|
|
|
|
2020-05-03 12:03:47 +02:00
|
|
|
XMLDictElement = namedtuple('XMLDictElement', ['tag', 'attrs', 'children'])
|
2020-04-28 08:46:30 +02:00
|
|
|
|
2020-04-30 03:01:37 +02:00
|
|
|
|
2020-05-03 12:03:47 +02:00
|
|
|
def xmlroot(tree):
|
2020-04-28 08:46:30 +02:00
|
|
|
""" Special process for root XML Node """
|
2020-05-03 12:03:47 +02:00
|
|
|
rootel = ET.Element(tree['tag'], tree['attrs'])
|
2020-04-30 03:01:37 +02:00
|
|
|
if 'children' in tree:
|
2020-05-03 12:03:47 +02:00
|
|
|
xmlchild(rootel, tree['children'])
|
2020-04-28 08:46:30 +02:00
|
|
|
return rootel
|
|
|
|
|
2020-05-03 12:03:47 +02:00
|
|
|
def xmlchild(parent, children):
|
2020-04-28 08:46:30 +02:00
|
|
|
""" Handling of children (ie non root) XML Nodes with/o text and
|
|
|
|
subchildren (recursive) """
|
2020-05-03 12:03:47 +02:00
|
|
|
if isinstance(children, str):
|
2020-04-30 03:01:37 +02:00
|
|
|
parent.text = children
|
2020-05-03 12:03:47 +02:00
|
|
|
elif isinstance(children, XMLDictElement):
|
|
|
|
attrs = {unicode(k): unicode(v) for [k, v] in children.attrs.items()}
|
|
|
|
new_parent = ET.SubElement(parent, children.tag, attrs)
|
|
|
|
subchildren = children.children
|
|
|
|
if subchildren:
|
|
|
|
xmlchild(new_parent, subchildren)
|
|
|
|
elif isinstance(children, list):
|
|
|
|
map(partial(xmlchild, parent), children)
|
|
|
|
else:
|
2020-04-30 10:58:36 +02:00
|
|
|
raise TypeError('Invalid arguments for xmlchild')
|
2020-04-28 08:46:30 +02:00
|
|
|
|
2020-05-03 12:03:47 +02:00
|
|
|
def xmln(tag='', attrs={}, children=[]):
|
2020-04-30 18:32:47 +02:00
|
|
|
""" XMLDictElement building from dict object, with defaults """
|
2020-05-03 12:03:47 +02:00
|
|
|
if isinstance(attrs, list):
|
2020-05-01 18:58:34 +02:00
|
|
|
children = attrs
|
|
|
|
attrs = {}
|
2020-05-03 12:03:47 +02:00
|
|
|
xmldictel = partial(XMLDictElement, tag, attrs)
|
|
|
|
if isinstance(children, str):
|
|
|
|
return xmldictel([children])
|
|
|
|
if isinstance(children, list):
|
|
|
|
return xmldictel(children)
|
|
|
|
raise TypeError('Invalid arguments for xmln')
|
2020-04-28 08:46:30 +02:00
|
|
|
|
2020-04-30 18:32:47 +02:00
|
|
|
|
2020-05-03 12:31:40 +02:00
|
|
|
def xml_write(filepath, tree, pretty=True):
|
2020-04-28 08:46:30 +02:00
|
|
|
""" Write XML file according to filename and given tree """
|
2020-05-03 12:03:47 +02:00
|
|
|
if filepath.endswith('.py'): # if .pyc, no need to generate XML
|
2020-05-03 12:31:40 +02:00
|
|
|
output_xml = ET.tostring(tree)
|
|
|
|
if pretty:
|
|
|
|
output_xml = minidom.parseString(output_xml).toprettyxml(indent=' ')
|
|
|
|
output_path = path.abspath(filepath).split(u'/')
|
|
|
|
output_path[-1] = output_path[-1].replace(u'.py', u'_views.xml')
|
|
|
|
output_path = u'/'.join(output_path)
|
|
|
|
with open(output_path, 'w') as output_file:
|
2020-04-29 11:02:18 +02:00
|
|
|
output_file.write(output_xml)
|