#!/usr/bin/env python # coding: utf-8 # # Creating Text-Fabric dataset (from GBI trees XML nodes) # # Version: 0.4 (July 24, 2023 - major updates; changing feature names; updated documentation) # # ## Table of content # * 1 - Introduction # * 2 - Read GBI XML data and store in pickle # * 2.1 - Import various libraries # * 2.2 - Initialize global data # * 2.3 - Function to add parent info to each node in XML tree # * 2.4 - Process the XML data and store in pickle file # * 3 - Nestle1904GBI Text-Fabric production from pickle input # * 3.1 - Load libraries and initialize some data # * 3.2 - Running the Text-Fabric walker function # # 1 - Introduction # ##### [Back to TOC](#TOC) # # The source data for the conversion are the XML node files representing the macula-greek version of Eberhard Nestle's 1904 Greek New Testament (British Foreign Bible Society, 1904). The starting dataset is formatted according to Syntax diagram markup by the Global Bible Initiative (GBI). The most recent source data can be found on github https://github.com/Clear-Bible/macula-greek/tree/main/Nestle1904/nodes. Attribution: "MACULA Greek Linguistic Datasets, available at https://github.com/Clear-Bible/macula-greek/". # # The production of the Text-Fabric files consist of two major parts. The first part is the creation of pickle files. The second part is the actual Text-Fabric creation process. Both parts are independent, allowing to start from part 2 by using the pickle files created in part 1 as input. # # # # # 2 - Read GBI XML data and store in pickle # ##### [Back to TOC](#TOC) # # This script extracts all information from the GBI tree XML data file, organizes it into a Pandas DataFrame, and saves the result per book in a pickle file. Please note that pickling in Python refers to the process of serializing an object into a disk file or buffer. See also the [Python3 documentation](https://docs.python.org/3/library/pickle.html). # # Within the context of this script, the term 'Leaf' refers to nodes that contain the Greek word as data. These nodes are also referred to as 'terminal nodes' since they do not have any children, similar to leaves on a tree. Additionally, Parent1 represents the parent of the leaf, Parent2 represents the parent of Parent1, and so on. For a visual representation, please refer to the following diagram. # # # # For a full description of the structure of the source data see document [MACULA Greek Treebank for the Nestle 1904 Greek New Testament.pdf](https://github.com/Clear-Bible/macula-greek/blob/main/doc/MACULA%20Greek%20Treebank%20for%20the%20Nestle%201904%20Greek%20New%20Testament.pdf) # ## 2.1 - Import various libraries # ##### [Back to TOC](#TOC) # In[2]: import pandas as pd import sys import os import time import pickle import re # used for regular expressions from os import listdir from os.path import isfile, join import xml.etree.ElementTree as ET # ## 2.2 - Initialize global data # ##### [Back to TOC](#TOC) # # IMPORTANT: To ensure proper creation of the Text-Fabric files on your system, it is crucial to adjust the values of BaseDir, InputDir, and OutputDir to match the location of the data and the operating system you are using. In this Jupyter Notebook, Windows is the operating system employed. # In[3]: BaseDir = 'C:\\Users\\tonyj\\my_new_Jupyter_folder\\test_of_xml_etree\\' InputDir = BaseDir+'inputfiles\\' OutputDir = BaseDir+'outputfiles\\' # key: filename, [0]=book_long, [1]=book_num, [3]=book_short bo2book = {'01-matthew': ['Matthew', '1', 'Matt'], '02-mark': ['Mark', '2', 'Mark'], '03-luke': ['Luke', '3', 'Luke'], '04-john': ['John', '4', 'John'], '05-acts': ['Acts', '5', 'Acts'], '06-romans': ['Romans', '6', 'Rom'], '07-1corinthians': ['I_Corinthians', '7', '1Cor'], '08-2corinthians': ['II_Corinthians', '8', '2Cor'], '09-galatians': ['Galatians', '9', 'Gal'], '10-ephesians': ['Ephesians', '10', 'Eph'], '11-philippians': ['Philippians', '11', 'Phil'], '12-colossians': ['Colossians', '12', 'Col'], '13-1thessalonians':['I_Thessalonians', '13', '1Thess'], '14-2thessalonians':['II_Thessalonians','14', '2Thess'], '15-1timothy': ['I_Timothy', '15', '1Tim'], '16-2timothy': ['II_Timothy', '16', '2Tim'], '17-titus': ['Titus', '17', 'Titus'], '18-philemon': ['Philemon', '18', 'Phlm'], '19-hebrews': ['Hebrews', '19', 'Heb'], '20-james': ['James', '20', 'Jas'], '21-1peter': ['I_Peter', '21', '1Pet'], '22-2peter': ['II_Peter', '22', '2Pet'], '23-1john': ['I_John', '23', '1John'], '24-2john': ['II_John', '24', '2John'], '25-3john': ['III_John', '25', '3John'], '26-jude': ['Jude', '26', 'Jude'], '27-revelation': ['Revelation', '27', 'Rev']} # ## 2.3 - Function to add parent info to each node in XML tree # ##### [Back to TOC](#TOC) # # To traverse from the 'leafs' (terminating nodes) up to the root of the tree, it is necessary to include information in each node that points to its parent. # # (The concept is derived from https://stackoverflow.com/questions/2170610/access-elementtree-node-parent-node) # In[4]: def addParentInfo(et): for child in et: child.attrib['parent'] = et addParentInfo(child) def getParent(et): if 'parent' in et.attrib: return et.attrib['parent'] else: return None # ## 2.4 - Process the XML data and store in pickle file # ##### [Back to TOC](#TOC) # This code processes books in the correct order. Firstly, it parses the XML and adds parent information to each node. Then, it loops through the nodes and checks if it is a 'leaf' node, meaning it contains only one word. If it is a 'leaf' node, the following steps are performed: # # * Adds computed data to the 'leaf' nodes in memory. # * Traverses from the 'leaf' node up to the root and adds information from the parent, grandparent, and so on, to the 'leaf' node. # * Once it reaches the root, it stops and stores all the gathered information in a dataframe that will be added to the full_dataframe. # * After processing all the nodes for a specific book, the full_dataframe is exported to a pickle file specific to that book. # # Note that this script takes a long time to execute (due to the large number of itterations). However, once the XML data is converted to PKL, there is no need to rerun (unless the source XML data is updated). # In[7]: # Set some globals monad=1 # Smallest meaningful unit of text (in this corpus: a single word) # Process all the books (files) in order for bo, bookinfo in bo2book.items(): CollectedItems=0 full_df=pd.DataFrame({}) book_long=bookinfo[0] booknum=bookinfo[1] book_short=bookinfo[2] InputFile = os.path.join(InputDir, f'{bo}.xml') OutputFile = os.path.join(OutputDir, f'{bo}.pkl') print(f'Processing {book_long} at {InputFile}') DataFrameList = [] # Send the loaded XML document to the parsing routine tree = ET.parse(InputFile) # Now add all the parent info to the nodes in the XML tree [this step is important!] addParentInfo(tree.getroot()) start_time = time.time() # Walk over all the leaves and harvest the data for elem in tree.iter(): if not list(elem): # If no child elements exist, this must be a leaf/terminal node # Show progress on screen by printing a dot for each 100 words processed CollectedItems+=1 if (CollectedItems%100==0): print (".",end='') # Leafref will contain list with book, chapter verse and wordnumber Leafref = re.sub(r'[!: ]'," ", elem.attrib.get('ref')).split() # Push value for monad to element tree elem.set('monad', monad) monad+=1 # Add some important computed data to the leaf elem.set('LeafName', elem.tag) elem.set('word', elem.text) elem.set('book_long', book_long) elem.set('booknum', int(booknum)) elem.set('book_short', book_short) elem.set('chapter', int(Leafref[1])) elem.set('verse', int(Leafref[2])) # The following code traces the parents up the tree and stores the discovered attributes. parentnode=getParent(elem) index=0 while (parentnode): index+=1 elem.set('Parent{}Name'.format(index), parentnode.tag) elem.set('Parent{}Type'.format(index), parentnode.attrib.get('Type')) elem.set('Parent{}Cat'.format(index), parentnode.attrib.get('Cat')) elem.set('Parent{}Start'.format(index), parentnode.attrib.get('Start')) elem.set('Parent{}End'.format(index), parentnode.attrib.get('End')) elem.set('Parent{}Rule'.format(index), parentnode.attrib.get('Rule')) elem.set('Parent{}Head'.format(index), parentnode.attrib.get('Head')) elem.set('Parent{}NodeId'.format(index),parentnode.attrib.get('nodeId')) elem.set('Parent{}ClType'.format(index),parentnode.attrib.get('ClType')) elem.set('Parent{}HasDet'.format(index),parentnode.attrib.get('HasDet')) currentnode=parentnode parentnode=getParent(currentnode) elem.set('parents', int(index)) # This will push all elements found in the tree into a DataFrame DataFrameChunk=pd.DataFrame(elem.attrib, index={monad}) DataFrameList.append(DataFrameChunk) # Store the resulting DataFrame per book into a pickle file for further processing full_df = pd.concat([df for df in DataFrameList]) output = open(r"{}".format(OutputFile), 'wb') pickle.dump(full_df, output) output.close() print("\nFound ",CollectedItems, " items in %s seconds\n" % (time.time() - start_time)) # ## 3 - Nestle1904GBI Text-Fabric production from pickle input # ##### [Back to TOC](#TOC) # # This script creates the Text-Fabric files by recursive calling the TF walker function. # API info: https://annotation.github.io/text-fabric/tf/convert/walker.html # # The pickle files created by step 1 are stored on Github location https://github.com/tonyjurg/Nestle1904GBI/tree/main/resources/picklefiles # ## 3.1 - Load libraries and initialize some data # ##### [Back to TOC](#TOC) # # Change BaseDir, InputDir and OutputDir to match location of the datalocation and the OS used. # In[8]: import pandas as pd import os import re import gc from tf.fabric import Fabric from tf.convert.walker import CV from tf.parameters import VERSION from datetime import date import pickle BaseDir = 'C:\\Users\\tonyj\\my_new_Jupyter_folder\\test_of_xml_etree\\' source_dir = BaseDir+'outputfiles\\' # the input for the walker is the output of the xml to pickle output_dir = BaseDir+'outputfilesTF\\' #the TextFabric files # key: filename, [0]=book_long, [1]=book_num, [3]=book_short bo2book = {'01-matthew': ['Matthew', '1', 'Matt'], '02-mark': ['Mark', '2', 'Mark'], '03-luke': ['Luke', '3', 'Luke'], '04-john': ['John', '4', 'John'], '05-acts': ['Acts', '5', 'Acts'], '06-romans': ['Romans', '6', 'Rom'], '07-1corinthians': ['I_Corinthians', '7', '1Cor'], '08-2corinthians': ['II_Corinthians', '8', '2Cor'], '09-galatians': ['Galatians', '9', 'Gal'], '10-ephesians': ['Ephesians', '10', 'Eph'], '11-philippians': ['Philippians', '11', 'Phil'], '12-colossians': ['Colossians', '12', 'Col'], '13-1thessalonians':['I_Thessalonians', '13', '1Thess'], '14-2thessalonians':['II_Thessalonians','14', '2Thess'], '15-1timothy': ['I_Timothy', '15', '1Tim'], '16-2timothy': ['II_Timothy', '16', '2Tim'], '17-titus': ['Titus', '17', 'Titus'], '18-philemon': ['Philemon', '18', 'Phlm'], '19-hebrews': ['Hebrews', '19', 'Heb'], '20-james': ['James', '20', 'Jas'], '21-1peter': ['I_Peter', '21', '1Pet'], '22-2peter': ['II_Peter', '22', '2Pet'], '23-1john': ['I_John', '23', '1John'], '24-2john': ['II_John', '24', '2John'], '25-3john': ['III_John', '25', '3John'], '26-jude': ['Jude', '26', 'Jude'], '27-revelation': ['Revelation', '27', 'Rev']} # ## 3.2 - Running the Text-Fabric walker function # ##### [Back to TOC](#TOC) # # Text-Fabric API info can be found at https://annotation.github.io/text-fabric/tf/convert/walker.html # # Explanatory notes about the data interpretation logic are incorporated within the Python code of the director function. # In[10]: TF = Fabric(locations=output_dir, silent=False) cv = CV(TF) ############################################### # Common helper functions # ############################################### # The following sanitizer function is required to prevent passing float data to the walker function def sanitize(input): if isinstance(input, float): return '' else: return (input) ############################################### # The director routine # ############################################### def director(cv): ############################################### # Innitial setup of data etc. # ############################################### NoneType = type(None) # needed as tool to validate certain data IndexDict = {} # init an empty dictionary for bo,bookinfo in bo2book.items(): ############################################### # start of section executed for each book # ############################################### # load all data into a dataframe and process books in order (note that bookinfo is a list) Book=bookinfo[0] BookNum=int(bookinfo[1]) BookShort=bookinfo[2] BookLoc = os.path.join(source_dir, f'{bo}.pkl') # Read data from PKL file and report progress print(f'\tloading {BookLoc}...') PklFile = open(BookLoc, 'rb') df_unsorted = pickle.load(PklFile) PklFile.close() # Fill dictionary of column names for this book ItemsInRow=1 for itemname in df_unsorted.columns.to_list(): IndexDict.update({'i_{}'.format(itemname): ItemsInRow}) # This is to identify the collumn containing the key to sort upon if itemname=="{http://www.w3.org/XML/1998/namespace}id": SortKey=ItemsInRow-1 ItemsInRow+=1 # Sort the nodes df=df_unsorted.sort_values(by=df_unsorted.columns[SortKey]) del df_unsorted # Reset/load the following initial variables (we are at the start of a new book) phrasefunction = prev_phrasefunction = phrasefunctionlong = prev_phrasefunctionlong='TBD' this_clausetype = this_clauserule = phrasetype="unknown" prev_chapter = prev_verse = prev_sentence = prev_clause = prev_phrase = int(1) sentence_track = clause_track = phrase_track = 1 sentence_done = clause_done = phrase_done = verse_done = chapter_done = book_done = False wrdnum = 0 # start at 0 # Create a set of nodes at the start a new book ThisBookPointer = cv.node('book') cv.feature(ThisBookPointer, book=Book, booknum=BookNum, bookshort=BookShort) ThisChapterPointer = cv.node('chapter') cv.feature(ThisChapterPointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=1) ThisVersePointer = cv.node('verse') cv.feature(ThisVersePointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=1, verse=1) ThisSentencePointer = cv.node('sentence') cv.feature(ThisSentencePointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=1, verse=1, sentence=1) ThisClausePointer = cv.node('clause') cv.feature(ThisClausePointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=1, verse=1, sentence=1, clause=1) ThisPhrasePointer = cv.node('phrase') cv.feature(ThisPhrasePointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=1, verse=1, sentence=1, clause=1, phrase=1) ############################################### # Iterate through words and construct objects # ############################################### for row in df.itertuples(): wrdnum += 1 # Get the number of parent nodes for this word parents = row[IndexDict.get("i_parents")] # Get chapter and verse for this word from the data chapter = row[IndexDict.get("i_chapter")] verse = row[IndexDict.get("i_verse")] # Get clause rule and type info of parent clause for i in range(1,parents-1): item = IndexDict.get("i_Parent{}Cat".format(i)) if row[item]=="CL": clauseparent=i this_clauserule=row[IndexDict.get("i_Parent{}Rule".format(i))] this_clausetype=row[IndexDict.get("i_Parent{}ClType".format(i))] break cv.feature(ThisClausePointer, clause=clause_track, clauserule=this_clauserule, clausetype=this_clausetype, book=Book, booknum=BookNum, bookshort=BookShort, chapter=chapter, verse=verse) # Get phrase type info prev_phrasetype=phrasetype for i in range(1,parents-1): item = IndexDict.get("i_Parent{}Cat".format(i)) if row[item]=="np": _item ="i_Parent{}Rule".format(i) phrasetype=row[IndexDict.get(_item)] break functionaltag=row[IndexDict.get('i_FunctionalTag')] # Determine syntactic categories of clause parts. See also the description in # "MACULA Greek Treebank for the Nestle 1904 Greek New Testament.pdf" page 5&6 # (section 2.4 Syntactic Categories at Clause Level) prev_phrasefunction=phrasefunction for i in range(1,clauseparent): phrasefunction = row[IndexDict.get("i_Parent{}Cat".format(i))] if phrasefunction=="ADV": phrasefunctionlong='Adverbial function' break elif phrasefunction=="IO": phrasefunctionlong='Indirect Object function' break elif phrasefunction=="O": phrasefunctionlong='Object function' break elif phrasefunction=="O2": phrasefunctionlong='Second Object function' break elif phrasefunction=="S": phrasefunctionlong='Subject function' break elif phrasefunction=='P': phrasefunctionlong='Predicate function' break elif phrasefunction=="V": phrasefunctionlong='Verbal function' break elif phrasefunction=="VC": phrasefunctionlong='Verbal Copula function' if prev_phrasefunction!=phrasefunction and wrdnum!=1: phrase_done = True # Determine syntactic categories at word level. See also the description in # "MACULA Greek Treebank for the Nestle 1904 Greek New Testament.pdf" page 6&7 # (2.2. Syntactic Categories at Word Level: Part of Speech Labels) sp=sanitize(row[IndexDict.get("i_Cat")]) if sp=='adj': splong='adjective' elif sp=='adj': splong='adjective' elif sp=='conj': splong='conjunction' elif sp=='det': splong='determiner' elif sp=='intj': splong='interjection' elif sp=='noun': splong='noun' elif sp=='num': splong='numeral' elif sp=='prep': splong='preposition' elif sp=='ptcl': splong='particle' elif sp=='pron': splong='pronoun' elif sp=='verb': splong='verb' ''' Determine if conditions are met to trigger some action action will be executed after next word ''' # Detect chapter boundary if prev_chapter != chapter: chapter_done = True verse_done=True sentence_done = True clause_done = True phrase_done = True # Detect verse boundary if prev_verse != verse: verse_done=True ''' Handle TF events and determine what actions need to be done if proper condition is met. ''' # Act upon end of phrase (close) if phrase_done or clause_done or sentence_done: cv.feature(ThisPhrasePointer, phrase=phrase_track, phrasetype=prev_phrasetype, phrasefunction=prev_phrasefunction, phrasefunctionlong=prev_phrasefunctionlong) cv.terminate(ThisPhrasePointer) prev_phrasefunction=phrasefunction prev_phrasefunctionlong=phrasefunctionlong # act upon end of clause (close) if clause_done: cv.terminate(ThisClausePointer) # act upon end of sentence (close) if sentence_done: cv.terminate(ThisSentencePointer) # act upon end of verse (close) if verse_done: cv.terminate(ThisVersePointer) prev_verse = verse # act upon end of chapter (close) if chapter_done: cv.terminate(ThisChapterPointer) prev_chapter = chapter # Start of chapter (create new) if chapter_done: ThisChapterPointer = cv.node('chapter') cv.feature(ThisChapterPointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=chapter) chapter_done = False # Start of verse (create new) if verse_done: ThisVersePointer = cv.node('verse') cv.feature(ThisVersePointer, book=Book, booknum=BookNum, bookshort=BookShort, chapter=chapter, verse=verse) verse_done = False # Start of sentence (create new) if sentence_done: ThisSentencePointer= cv.node('sentence') cv.feature(ThisSentencePointer, sentence=sentence_track) sentence_track += 1 sentence_done = False # Start of clause (create new) if clause_done: ThisClausePointer = cv.node('clause') cv.feature(ThisClausePointer, clause=clause_track, clauserule=this_clauserule,clausetype=this_clausetype) clause_track += 1 clause_done = False phrase_done = True # Start of phrase (create new) if phrase_done: ThisPhrasePointer = cv.node('phrase') cv.feature(ThisPhrasePointer, phrase=phrase_track, phrasefunction=phrasefunction, phrasefunctionlong=phrasefunctionlong) prev_phrase = phrase_track prev_phrasefunction=phrasefunction prev_phrasefunctionlong=phrasefunctionlong phrase_track += 1 phrase_done = False # Detect boundaries of sentences, clauses and phrases text=row[IndexDict.get("i_Unicode")] if text[-1:] == "." : sentence_done = True clause_done = True phrase_done = True if text[-1:] == ";" or text[-1:] == ",": clause_done = True phrase_done = True ''' -- create word nodes -- ''' # Get the word details and detect presence of punctuations word=row[IndexDict.get("i_Unicode")] match = re.search(r"([\.·—,;])$", word) if match: # The group(0) method is used to retrieve the matched punctuation sign after=match.group(0)+' ' # Remove the punctuation from the end of the word word=word[:-1] else: after=' ' # Some attributes are not present inside some (small) books. The following is to prevent exceptions. degree='' if 'i_Degree' in IndexDict: degree=sanitize(row[IndexDict.get("i_Degree")]) subjref='' if 'i_SubjRef' in IndexDict: subjref=sanitize(row[IndexDict.get("i_SubjRef")]) # Create the word node ThisWordPointer = cv.slot() cv.feature(ThisWordPointer, after=after, word=word, monad=row[IndexDict.get("i_monad")], book=Book, booknum=BookNum, bookshort=BookShort, chapter=chapter, sp=sp, splong=splong, verse=verse, sentence=sentence_track, clause=clause_track, phrase=phrase_track, normalized=sanitize(row[IndexDict.get("i_NormalizedForm")]), formaltag=sanitize(row[IndexDict.get("i_FormalTag")]), functionaltag=functionaltag, strongs=sanitize(row[IndexDict.get("i_StrongNumber")]), lex_dom=sanitize(row[IndexDict.get("i_LexDomain")]), ln=sanitize(row[IndexDict.get("i_LN")]), gloss=sanitize(row[IndexDict.get("i_Gloss")]), gn=sanitize(row[IndexDict.get("i_Gender")]), nu=sanitize(row[IndexDict.get("i_Number")]), case=sanitize(row[IndexDict.get("i_Case")]), lemma=sanitize(row[IndexDict.get("i_UnicodeLemma")]), person=sanitize(row[IndexDict.get("i_Person")]), mood=sanitize(row[IndexDict.get("i_Mood")]), tense=sanitize(row[IndexDict.get("i_Tense")]), number=sanitize(row[IndexDict.get("i_Number")]), voice=sanitize(row[IndexDict.get("i_Voice")]), degree=degree, type=sanitize(row[IndexDict.get("i_Type")]), reference=sanitize(row[IndexDict.get("i_Ref")]), # the capital R is critical here! subj_ref=subjref, nodeID=row[1] #this is a fixed position. ) cv.terminate(ThisWordPointer) ''' Wrap up the book. At the end of the book we need to close all nodes in proper order. ''' # Close all nodes (phrase, clause, sentence, verse, chapter and book) cv.feature(ThisPhrasePointer, phrase=phrase_track, phrasetype=prev_phrasetype,phrasefunction=prev_phrasefunction,phrasefunctionlong=prev_phrasefunctionlong) cv.terminate(ThisPhrasePointer) cv.feature(ThisClausePointer, clause=clause_track, clauserule=this_clauserule, clausetype=this_clausetype) cv.terminate(ThisClausePointer) cv.terminate(ThisSentencePointer) cv.terminate(ThisVersePointer) cv.terminate(ThisChapterPointer) cv.terminate(ThisBookPointer) # Clear dataframe for this book del df # Clear the index dictionary IndexDict.clear() gc.collect() ############################################### # End of section executed for each book # ############################################### ############################################### # End of director function # ############################################### ############################################### # Output definitions # ############################################### slotType = 'word' # or whatever you choose otext = { # dictionary of config data for sections and text formats 'fmt:text-orig-full':'{word}{after}', 'sectionTypes':'book,chapter,verse', 'sectionFeatures':'book,chapter,verse', 'structureFeatures': 'book,chapter,verse', 'structureTypes': 'book,chapter,verse', } # Configure metadata generic = { # dictionary of metadata which will be included in all feature files 'TextFabric version': '{}'.format(VERSION), #imported from tf.parameter 'XML source location': 'https://github.com/tonyjurg/Nestle1904GBI/tree/main/resources/sourcedata/apr_6_2023', 'XML source date': 'April 6, 2023', 'Author': 'Evangelists and apostles', 'Availability': 'Creative Commons Attribution 4.0 International (CC BY 4.0)', 'Converter author': 'Tony Jurg', 'Converter execution': 'Tony Jurg', 'Convertor source': 'https://github.com/tonyjurg/Nestle1904GBI/tree/main/resources/converter', 'Converter version': '0.4', 'Data source': 'MACULA Greek Linguistic Datasets, available at https://github.com/Clear-Bible/macula-greek/tree/main/Nestle1904/nodes', 'Editors': 'Eberhart Nestle (1904)', 'Source description': 'Greek New Testment (British Foreign Bible Society, 1904)', 'Source format': 'XML (GBI tree node data)', 'Title': 'Greek New Testament (Nestle 1904 GBI)' } intFeatures = { # set of integer valued feature names 'booknum', 'chapter', 'verse', 'sentence', 'clause', 'phrase', 'monad' } featureMeta = { # per feature dicts with metadata 'after': {'Description': 'Chararcter after the word (space or punctuation)'}, 'book': {'Description': 'Book name (fully spelled out)'}, 'booknum': {'Description': 'NT book number (Matthew=1, Mark=2, ..., Revelation=27)'}, 'bookshort': {'Description': 'Book name (abbreviated)'}, 'chapter': {'Description': 'Chapter number inside book'}, 'verse': {'Description': 'Verse number inside chapter'}, 'sentence': {'Description': 'Sentence number (counted per chapter)'}, 'clause': {'Description': 'Clause number (counted per chapter)'}, 'clauserule': {'Description': 'Clause rule'}, 'clausetype': {'Description': 'Clause type'}, 'phrase' : {'Description': 'Phrase number (counted per chapter)'}, 'phrasetype' : {'Description': 'Phrase type information'}, 'phrasefunction' : {'Description': 'Phrase function (abbreviated)'}, 'phrasefunctionlong' : {'Description': 'Phrase function (long description)'}, 'monad': {'Description': 'Sequence number of the smallest meaningful unit of text (single word)'}, 'word': {'Description': 'Word as it appears in the text'}, 'sp': {'Description': 'Speech Part (abbreviated)'}, 'splong': {'Description': 'Speech Part (long description)'}, 'normalized': {'Description': 'Surface word stripped of punctations'}, 'lemma': {'Description': 'Lexeme (lemma)'}, 'formaltag': {'Description': 'Formal tag (Sandborg-Petersen morphology)'}, 'functionaltag': {'Description': 'Functional tag (Sandborg-Petersen morphology)'}, # see also discussion on relation between lex_dom and ln @ https://github.com/Clear-Bible/macula-greek/issues/29 'lex_dom': {'Description': 'Lexical domain according to Semantic Dictionary of Biblical Greek, SDBG'}, 'ln': {'Description': 'Lauw-Nida lexical classification'}, 'strongs': {'Description': 'Strongs number'}, 'gloss': {'Description': 'English gloss'}, 'gn': {'Description': 'Gramatical gender (Masculine, Feminine, Neuter)'}, 'nu': {'Description': 'Gramatical number (Singular, Plural)'}, 'case': {'Description': 'Gramatical case (Nominative, Genitive, Dative, Accusative, Vocative)'}, 'person': {'Description': 'Gramatical person of the verb (first, second, third)'}, 'mood': {'Description': 'Gramatical mood of the verb (passive, etc)'}, 'tense': {'Description': 'Gramatical tense of the verb (e.g. Present, Aorist)'}, 'number': {'Description': 'Gramatical number of the verb'}, 'voice': {'Description': 'Gramatical voice of the verb'}, 'degree': {'Description': 'Degree (e.g. Comparitative, Superlative)'}, 'type': {'Description': 'Gramatical type of noun or pronoun (e.g. Common, Personal)'}, 'reference': {'Description': 'Reference (to nodeID in XML source data, not yet post-processes)'}, 'subj_ref': {'Description': 'Subject reference (to nodeID in XML source data)'}, 'nodeID': {'Description': 'Node ID (as in the XML source data)'} } ''' -- The main function -- ''' good = cv.walk( director, slotType, otext=otext, generic=generic, intFeatures=intFeatures, featureMeta=featureMeta, warn=True, force=True ) if good: print ("done") # In[ ]: