Parsing milestoned XML in Python ================================ :date: 2016-08-25T22:13:29 :category: computer :tags: python, xml, Bible, recursive, generator .. zotero-setup:: :style: chicago-author-date I am trying to write a tool in Python (using Python 3.4 to be able to use the latest Python standard library on Windows without using any external libraries on Windows) for some manipulation with the source code for the Bible texts. Let me first explain what is the milestoned XML, because many normal Python programmers dealing with normal XML documents may not be familiar with it. There is a problem with using XML markup for documents with complicated structure. One rather complete article on this topic is :xcite:`@derose:proceedings`. Briefly [#]_ , the problem in many areas (especially in documents processing) is with multiple possible hierarchies overlapping each other (e.g., in Bibles there are divisions of text which are going across verse and chapters boundaries and sometimes terminating in the middle of verse, many especially English Bibles marks Jesus’ sayings with a special element, and of course this can go over several verses etc.). One of the ways how to overcome obvious problem that XML doesn't allow overlapping elements is to use milestones_. So for example the book of Bible could be divided not like .. code-block:: xml text ... ... but just putting milestones in the text, i.e.: .. code-block:: xml text of verse 1.1 .... So, in my case the part of the document may look like .. code-block:: xml text text textB textB textC textC textD textD And I would like to get from some kind of iterator this series of outputs: .. code-block:: python [(1, 1, "text text", ['text text']), (1, 2, "textB textB textC textC", ['', 'textB textB', '', 'textC textC']), (1, 3, "textD textD", ['', 'textD textD', ''])] (the first two numbers should be number of the chapter and verse respectively). My first attempt was in its core this iterator: .. code-block:: python def __iter__(self) -> Tuple[int, int, str]: """ iterate through the first level elements NOTE: this iterator assumes only all milestoned elements on the first level of depth. If this assumption fails, it might be necessary to rewrite this function (or perhaps ``text`` method) to be recursive. """ collected = None for child in self.root: if child.tag in ['titulek']: continue if child.tag in ['kap', 'vers']: if collected and collected.strip(): yield self.cur_chapter, self.cur_verse, \ self._list_to_clean_text(collected) if child.tag == 'kap': self.cur_chapter = int(child.get('n')) elif child.tag == 'vers': self.cur_verse = int(child.get('n')) collected = child.tail or '' else: if collected is not None: if child.text is not None: collected += child.text for sub_child in child: collected += self._recursive_get_text(sub_child) if child.tail is not None: collected += child.tail (``self.root`` is a product of ``ElementTree.parse(file_name).getroot()``). The problem of this code lies in the note. When the ```` element is inside of ```` one, it is ignored. So, obviously we have to make our iterator recursive. My first idea was to make this script parsing and regenerating XML: .. code-block:: python #!/usr/bin/env python3 from xml.etree import ElementTree as ET from typing import List def start_element(elem: ET.Element) -> str: outx = ['<{} '.format(elem.tag)] for attr, attval in elem.items(): outx.append('{}={} '.format(attr, attval)) outx.append('>') return ''.join(outx) def recursive_parse(elem: ET.Element) -> List[str]: col_xml = [] col_txt = '' cur_chapter = chap if elem.text is None: col_xml.append(ET.tostring(elem)) if elem.tail is not None: col_txt += elem.tail else: col_xml.extend([start_element(elem), elem.text]) col_txt += elem.text for subch in elem: subch_xml, subch_text = recursive_parse(subch) col_xml.extend(subch_xml) col_txt += subch_text col_xml.append(''.format(elem.tag)) if elem.tail is not None: col_xml.append(elem.tail) col_txt += elem.tail return col_xml, col_txt if __name__ == '__main__': # write result XML to CRLF-delimited file with # ET.tostring(ET.fromstringlist(result), encoding='utf8') # or encoding='unicode'? Better for testing? xml_file = ET.parse('tests/data/Mat-old.xml') collected_XML, collected_TEXT = recursive_parse(xml_file.getroot()) with open('test.xml', 'w', encoding='utf8', newline='\r\n') as outf: print(ET.tostring(ET.fromstringlist(collected_XML), encoding='unicode'), file=outf) with open('test.txt', 'w', encoding='utf8', newline='\r\n') as outf: print(collected_TEXT, file=outf) This works correctly in sense that the generated file ``test.xml`` is identical to the original XML file (after reformatting both files with ``tidy -i -xml -utf8``). However, it is not iterator, so I would like to somehow combine the virtues of both snippets of code into one. Obviously, the problem is that ``return`` in my ideal code should serve two purposes. Once it should actually yield nicely formatted result from the iterator, second time it should just provide content of the inner elements (or not, if the inner element contains ```` element). If my ideal world I would like to get ``recursive_parse()`` to function as an iterator capable of something like this: .. code-block:: python if __name__ == '__main__': xml_file = ET.parse('tests/data/Mat-old.xml') parser = ET.XMLParser(target=ET.TreeBuilder()) with open('test.txt', 'w', newline='\r\n') as out_txt, \ open('test.xml', 'w', newline='\r\n') as out_xml: for ch, v, verse_txt, verse_xml in recursive_parse(xml_file): print(verse_txt, file=out_txt) # or directly parser.feed(verse_xml) # if verse_xml is not a list parser.feed(''.join(verse_xml)) print(ET.tostring(parser.close(), encoding='unicode'), file=out_xml) So, my first attempt to rewrite the iterator (so far without the XML part I have): .. code-block:: python def __iter__(self) -> Tuple[CollectedInfo, str]: """ iterate through the first level elements """ cur_chapter = 0 cur_verse = 0 collected_txt = '' # collected XML is NOT directly convertable into Element objects, # it should be treated more like a list of SAX-like events. # # xml.etree.ElementTree.fromstringlist(sequence, parser=None) # Parses an XML document from a sequence of string fragments. # sequence is a list or other sequence containing XML data fragments. # parser is an optional parser instance. If not given, the standard # XMLParser parser is used. Returns an Element instance. # # sequence = ["", "text"] # element = ET.fromstringlist(sequence) # self.assertEqual(ET.tostring(element), # b'text') # FIXME přidej i sběr XML útržků # collected_xml = None for child in self.root.iter(): if child.tag in ['titulek']: collected_txt += '\n{}\n'.format(child.text) collected_txt += child.tail or '' if child.tag in ['kap', 'vers']: if collected_txt and collected_txt.strip(): yield CollectedInfo(cur_chapter, cur_verse, re.sub(r'[\s\n]+', ' ', collected_txt, flags=re.DOTALL).strip()), \ child.tail or '' if child.tag == 'kap': cur_chapter = int(child.get('n')) elif child.tag == 'vers': cur_verse = int(child.get('n')) else: collected_txt += child.text or '' for sub_child in child: for sub_info, sub_tail in MilestonedElement(sub_child): if sub_info.verse == 0 or sub_info.chap == 0: collected_txt += sub_info.text + sub_tail else: # FIXME what happens if sub_element contains # multiple elements? yield CollectedInfo( sub_info.chap, sub_info.verse, collected_txt + sub_info.text), '' collected_txt = sub_tail collected_txt += child.tail or '' yield CollectedInfo(0, 0, collected_txt), '' Am I going the right way, or did I still not get it? .. [#] From the discussion_ of the topic on the XSL list. .. _discussion: http://www.oxygenxml.com/archives/xsl-list/201202/msg00170.html .. _milestones: https://wiki.crosswire.org/OSIS_Bibles#OSIS_Milestones .. bibliography::