home.lib.jython.alertlib.Alert.py Maven / Gradle / Ivy
from CSVAlertHandler import CSVAlertHandler
from XMLAlertHandler import XMLAlertHandler
from AlertException import AlertException
import os
import traceback
import time
import shutil
from com.googlecode.fascinator import HarvestClient
from com.googlecode.fascinator.common import FascinatorHome
from java.lang import Exception
from java.io import File
from org.apache.commons.io import FileUtils
from org.apache.commons.lang.text import StrSubstitutor
class Alert:
'''
This is the main alert handling class - each instance represents one alert (i.e. an element in the alert-set array)
'''
#This is usually only changed in testing via:
# mock.patch('Alert.Alert.debug', True)
debug = False
def __init__(self, redboxVersion, config, baseline, log):
self.config = config
self.redboxVersion = redboxVersion
self.name = config['name']
self.path = StrSubstitutor.replaceSystemProperties(config['path'])
self.harvestConfig = StrSubstitutor.replaceSystemProperties(config['harvestConfig'])
self.handlers = config['handlers']
self.processLog = []
self.__log = log
if 'baseline' in config:
#Merge the alert's fixed metadata over the main baseline
self.baseline = dict(baseline.items() + config['baseline'].items())
else:
self.baseline = baseline
self.timestampFields = []
if 'timestampFields' in config:
self.timestampFields = config['timestampFields']
#These directories are used to hold files during/after processing
self.__DIR_PROCESSED = os.path.join(self.path, ".processed")
self.__DIR_ALERT = os.path.join(self.__DIR_PROCESSED, time.strftime("%Y_%m_%d_%H_%M_%S"))
self.__DIR_PROCESSING = os.path.join(self.__DIR_PROCESSED, "processing/")
self.__DIR_SUCCESS = os.path.join(self.__DIR_ALERT, "success/")
self.__DIR_FAILED = os.path.join(self.__DIR_ALERT, "failed/")
self.__DIR_ORIGINAL = os.path.join(self.__DIR_ALERT, "original/")
def processAlert(self):
'''Undertakes the alert handling
The Alert is handled as "quietly" as possible. Exceptions etc are logged rather than thrown (where possible).
Logs and processing files are written to a ".processed" directory in the path of the alert.
Exceptions:
Exception - thrown if the directories needed to handle processing don't exist or can't be created
'''
try:
self.__checkDirs()
except Exception, e:
#We can't handle this alert as the dirs weren't available
raise
files = os.listdir(self.path)
for file in files:
if not os.path.isfile(self.pBase(file)):
#Ignore sub-dirs
continue
try:
success = 0
fail = 0
#Start a new processLog for each alert
self.processLog = []
try:
(success, fail) = self.__handleAlert(file)
self.logInfo(file, "File processing complete. Successful imports: %s; Failed imports %s"%(success,fail))
except AlertException, e:
self.__log.error("The alert [%s] was handling [%s] and raised an exception: %s" % (self.name, file, e.message))
self.logException(file, e)
finally:
logFile = os.path.join(self.__DIR_ALERT, file + ".log")
self.saveAlertLog(logFile)
self.__log.info("Log file created: %s" % logFile)
#Move the original file to an archive folder
self.__log.info("Moving %s to %s" % (self.pBase(file), os.path.join(self.__DIR_ORIGINAL,file)))
shutil.move(self.pBase(file), os.path.join(self.__DIR_ORIGINAL,file))
return
def __handleAlert(self, file):
'''Sends the alert off to the relevant handler and then pushes the metadata to ReDBox
Parameters:
file -- the file (path, not object) to be processed
'''
successCount = 0
failedCount = 0
handler = None
self.logInfo(file, "Processing file " + file)
self.__log.info("Alert system is processing file %s" % file)
ext = file.rpartition('.')[2]
if not ext in self.handlers:
self.logInfo(file, "Did not process file as extension is not configured")
return (0,0)
#Add the timestamp to a copy of baseline
baseline = dict(self.baseline)
timestamp = time.gmtime(os.path.getmtime(self.pBase(file)))
for field in self.timestampFields:
val = time.strftime("%Y-%m-%d %H:%M:%S", timestamp)
baseline[field] = val
if self.handlers[ext] == "CSVAlertHandler":
config = self.config['CSVAlertHandlerParams']['configMap'][ext]
handler = CSVAlertHandler(self.pBase(file), config, baseline)
self.logInfo(file, "Using the CSVAlertHandler for file with extension %s" % ext)
elif self.handlers[ext] == "XMLAlertHandler":
config = self.config['XMLAlertHandlerParams']['configMap'][ext]
handler = XMLAlertHandler(self.pBase(file), config, baseline)
self.logInfo(file, "Using the XMLAlertHandler for file with extension %s" % ext)
else:
raise AlertException("Unknown file handler: '%s'" % self.handlers[ext])
jsonList = handler.process()
if jsonList is None:
self.logInfo(file, "No records were returned.")
return(0,0)
## Now all of the JSON Objects need to be ingested into the tool chain
id = 0
for json in jsonList:
id += 1
#use an incremental filename in case the data file contains more than 1 record
meta_file_name = "%s.%s" % (file,id)
meta_file = self.pTemp(meta_file_name)
self.logInfo(file, "Using metadata file: %s" % meta_file)
try:
oid = self.__ingestJson(file, meta_file, json)
successCount += 1
except Exception, e:
failedCount += 1
self.logInfo(file, "Moving failed metadata file [%s] to %s." % (meta_file, self.__DIR_FAILED))
filepath = os.path.join(self.__DIR_FAILED,meta_file_name)
FileUtils.moveFile(File(meta_file),File(filepath));
continue
self.logInfo(file, "Moving successful metadata file [%s] to %s." % (meta_file, self.__DIR_SUCCESS))
filepath = os.path.join(self.__DIR_SUCCESS,meta_file_name)
#python library seems to dislike mixed \ and / in path
FileUtils.moveFile(File(meta_file),File(filepath));
return (successCount, failedCount)
def __ingestJson(self, file, meta_file, json):
'''Takes a json object and sends it through to the harvester
file -- the source file
meta_file -- the name to use for the resulting source file of the data
json -- the json construct to be sent to the harvester
'''
harvester = None
oid = None
#self.logInfo(file, "%s" % json.toString(True))
## Cache the file out to disk... although it requires
## .tfpackage extension due to jsonVelocity transformer
jsonFile = None
try:
jsonFile = open(meta_file, "wb")
jsonFile.write(json.toString(True))
jsonFile.close()
except Exception, e:
raise
finally:
if jsonFile is not None:
jsonFile.close()
#Determine the object's owner from the alert config
#owner = "guest"
#if self.config("owner"):
#pass
self.logInfo(file, "Submitting to harvest. Config file is %s and meta_file is %s" % (self.harvestConfig, meta_file))
try:
## Now instantiate a HarvestClient just for this File.
#harvester = HarvestClient(File(self.harvestConfig), File(meta_file), owner)
harvester = HarvestClient(File(self.harvestConfig), File(meta_file), None)
harvester.start()
## And cleanup afterwards
oid = harvester.getUploadOid()
except Exception, e:
self.logException(file, e)
raise
finally:
## Cleanup
if harvester is not None:
harvester.shutdown()
self.logInfo(file, "Successfully harvested alert item %s from processing file %s" % (oid,meta_file))
return oid
def __checkDirs(self):
# Makes sure that the required directories exist: .processed, failed, processed, success, original
# All alert directories will have 1 process folder: .processed
self.__createDir(self.__DIR_PROCESSED)
#Under that directory will be a subdir for each of the alerts run
self.__createDir(self.__DIR_ALERT)
#And finally we have the processing sub-subdirs
self.__createDir(self.__DIR_PROCESSING)
self.__createDir(self.__DIR_SUCCESS)
self.__createDir(self.__DIR_FAILED)
self.__createDir(self.__DIR_ORIGINAL)
return
def __createDir(self, dir):
self.__log.debug("Checking directory: %s" % dir)
try:
dir = dir.replace('\\','/')
os.mkdir(dir)
except OSError:
#Thrown when directory already exists so ignore
pass
if not os.path.exists(dir):
self.__log.error("Required processing directory %s does not exist. I even tried to create it for you." % dir)
raise AlertException("Required processing directory %s does not exist. I even tried to create it for you." % dir)
# Short nameed Wrappers for convention based file paths
def pBase(self, file):
# A base path file
return os.path.join(self.path, file)
def pTemp(self, file):
# Cache file for the HarvestClient... '.tfpackage' extension
# required inside system due to jsonVelocity transformer
return os.path.join(self.__DIR_PROCESSING, file + ".tfpackage")
def log(self, type, fileName, message):
self.processLog.append( {"timestamp": time.ctime(),
"type": type,
"alert": self.name,
"file": fileName,
"message": message})
if self.debug:
print "%s\t%s\t%s\t%s" % (type, self.name, fileName, message)
def logException (self, fileName, e):
self.log("ERROR", fileName, "Message was: [%s] Exception was: [%s]" % (e.message, traceback.format_exc()))
def logInfo(self, fileName, message):
self.log("INFO", fileName, message)
def saveAlertLog(self, file):
f = None
try:
f = open(file, "wb")
for item in self.processLog:
if item["type"] == "ERROR":
f.write("[ERROR]\t[%s]\t[%s]\t[%s]\t%s\n" % (item["alert"], item["file"], item["timestamp"], item["message"]))
else:
f.write("[INFO] [%s]\t[%s]\t[%s]\t%s\n" % (item["alert"], item["file"], item["timestamp"], item["message"]))
finally:
if not f is None:
f.close()
© 2015 - 2025 Weber Informatics LLC | Privacy Policy