All Downloads are FREE. Search and download functionalities are using the official Maven repository.

home.lib.jython.alertlib.XMLAlertHandler.py Maven / Gradle / Ivy

There is a newer version: 1.10.2
Show newest version
from AlertException import AlertException
from AlertHandler import AlertHandler

from org.dom4j import DocumentFactory
from org.dom4j.io import SAXReader
from org.dom4j.xpath import DefaultNamespaceContext
from org.dom4j.xpath import DefaultXPath
from org.jaxen import SimpleNamespaceContext

from com.googlecode.fascinator.common import JsonObject
from com.googlecode.fascinator.common import JsonSimple

from org.json.simple import JSONArray

from java.io import File
from java.io import FileInputStream
from java.io import InputStreamReader
from java.lang import Exception
from org.apache.commons.lang.text import StrSubstitutor

import os

class XMLAlertHandler(AlertHandler):
    ''''Processing class for a single XML File.
    Each XML file is expected to contain only a single Collection
    '''

    def __init__(self, file, config, baseline):
        AlertHandler.__init__(self, file, config, baseline)
        docFactory = DocumentFactory()
        self.saxReader = SAXReader(docFactory)
        self.xmlMapFile = StrSubstitutor.replaceSystemProperties(config['xmlMap'])
        if not os.path.exists(self.xmlMapFile):
            raise AlertException("Requested xmlMap file %s does not exist." % self.xmlMapFile)
        
        ## Make sure we can see our mappings
        inStream = FileInputStream(File(self.xmlMapFile))
        xmlMappings = JsonSimple(inStream)
        self.map = xmlMappings.getObject(["mappings"])
        self.exceptions = xmlMappings.getObject(["exceptions"])
        self.defaultNamespace = xmlMappings.getObject(["defaultNamespace"])
        
        self.mappedExceptionCount = 0
        
        
    def process(self):
        '''Read the XML file and map xpath items to metadata
        Return a list with 1 JsonSimple object (at most)
        '''
        jsonList = []
        data = None
        reader = None
        inStream = None
        document = None
        
        # Run the XML through our parser
        try:
            inStream = FileInputStream(File(self.file))
            reader = InputStreamReader(inStream, "UTF-8")
            document = self.saxReader.read(reader)
        # Parse fails
        except:
            raise
        # Close our file access objects
        finally:
            if reader is not None:
                reader.close()
            if inStream is not None:
                inStream.close()

        # Now go looking for all our data
        data = self.getNewJsonObject()
        self.__mapXpathToFields(document, self.map, data)
        
        if data is None:
            return None
        
        jsonList.append(JsonSimple(data))
        return jsonList


    ## Used recursively
    def __mapXpathToFields(self, sourceData, map, responseData, index = 1):
        for xpath in map.keySet():
            field = map.get(xpath)
            if xpath != "":
                xpathobj = DefaultXPath(xpath)
                if not self.defaultNamespace is None:
                    xpathobj.setNamespaceContext(SimpleNamespaceContext(self.defaultNamespace))
                    
                nodes = xpathobj.selectNodes(sourceData)
                
                if isinstance(field, JsonObject):
                    #The XPath key provides a dictionary containing sub xpath queries mapped to fields
                    i = 1
                    for node in nodes:
                        self.__mapXpathToFields(node, field, responseData, i)
                        i += 1
                else:
                    # Lists indicate we're copying the several fields
                    if isinstance(field, JSONArray):
                        for eachField in field:
                            self.__insertFieldData(nodes, eachField, responseData, index)
                    # or just one field
                    else:
                        self.__insertFieldData(nodes, field, responseData, index)

    def __insertFieldData(self, xmlNodes, field, responseData, index):
        multiValue = False
        multiIndex = 1
        fieldString = ""
        
        if self.exceptions["fields"].containsKey(field):
            #The field is an exception
            excepted = True
            output = self.exceptions["output"]
            self.mappedExceptionCount += 1
        else:
            # Nope, just normal
            excepted = False
            if ('.0.' in field and len(xmlNodes) > 1): 
            #In ReDBox, a field such as dc:subject.vivo:keyword.0.rdf:PlainLiteral indicates a list of values, using the number as a counter.
            #In the code below, if a field contains this number element, we can increment the counter and add more and more. 
            #If there is no number, we just overwrite the value.
                multiValue = True
                #we'll do the fieldString index change a little later
                fieldString = field
            else:
                fieldString = field.replace(".0.", ".%s."%index, 1)
            
        for node in xmlNodes:
            try:
                text = node.getTextTrim()
            except:
            	try:
                    text = node.getValue().strip()
                except:
                    text = node
                
            if fieldString != "" and text != "":
                if excepted:
                    exceptionString = "%s: '%s' (%s)" % (exceptions["fields"][field], text, field)
                    responseData.put(fieldString, exceptionString)
                else:
                    if multiValue:
                        fieldString = field.replace(".0.", ".%s."%multiIndex, 1)
                        multiIndex += 1
                    responseData.put(fieldString, text)
    




© 2015 - 2025 Weber Informatics LLC | Privacy Policy