diff --git a/config.yaml b/config.yaml index 13f35c3..b1572fb 100644 --- a/config.yaml +++ b/config.yaml @@ -115,6 +115,8 @@ database: username: dbeaver password: password database: hss2 + readCacheEnabled: True + readCacheInterval: 60 ## External Webhook Notifications webhooks: diff --git a/lib/banners.py b/lib/banners.py index 933a361..8ac23b7 100755 --- a/lib/banners.py +++ b/lib/banners.py @@ -88,5 +88,23 @@ def logService(self) -> str: Log Service +""" + return bannerText + + def databaseService(self) -> str: + bannerText = """ + + ###### ## ## ##### ##### + ## ## ## ## ## ## ## ## + ## ## ## ## ## ## ## ## + ###### ## ## ####### ##### ##### + ## ## ## ## ## ## ## + ## ## ## ## ## ## ## ## ## + ## ##### ## ## ##### ##### + ## + #### + + Database Service + """ return bannerText \ No newline at end of file diff --git a/services/databaseService.py b/services/databaseService.py new file mode 100644 index 0000000..d353a7c --- /dev/null +++ b/services/databaseService.py @@ -0,0 +1,143 @@ +import os, sys, json, yaml +import uuid, time +import asyncio +import socket +import datetime +import traceback +sys.path.append(os.path.realpath('../lib')) +from messagingAsync import RedisMessagingAsync +from banners import Banners +from logtool import LogTool +from sqlalchemy import create_engine, inspect +from sqlalchemy.orm import sessionmaker +from sqlalchemy import MetaData, Table + +class DatabaseService: + """ + Redis-Database Cache Service + Functions as an asynchronous cache for a database. + Currently read-only. + """ + + def __init__(self, redisHost: str='127.0.0.1', redisPort: int=6379): + try: + with open("../config.yaml", "r") as self.configFile: + self.config = yaml.safe_load(self.configFile) + except: + print(f"[Database] Fatal Error - config.yaml not found, exiting.") + quit() + self.logTool = LogTool(self.config) + self.banners = Banners() + + self.redisUseUnixSocket = self.config.get('redis', {}).get('useUnixSocket', False) + self.redisUnixSocketPath = self.config.get('redis', {}).get('unixSocketPath', '/var/run/redis/redis-server.sock') + self.redisHost = self.config.get('redis', {}).get('host', 'localhost') + self.redisPort = self.config.get('redis', {}).get('port', 6379) + self.redisDatabaseReadMessaging = RedisMessagingAsync(host=self.redisHost, port=self.redisPort, useUnixSocket=self.redisUseUnixSocket, unixSocketPath=self.redisUnixSocketPath) + self.redisLogMessaging = RedisMessagingAsync(host=self.redisHost, port=self.redisPort, useUnixSocket=self.redisUseUnixSocket, unixSocketPath=self.redisUnixSocketPath) + self.hostname = socket.gethostname() + + supportedDatabaseTypes = ['mysql'] + self.databaseType = self.config.get('database', {}).get('db_type', 'mysql').lower() + if not self.databaseType in supportedDatabaseTypes: + print(f"[Database] Fatal Error - unsupported database type: {self.databaseType}. Supported database types are: {supportedDatabaseTypes}, exiting.") + quit() + + self.databaseHost = self.config.get('database', {}).get('server', '') + self.databaseUsername = self.config.get('database', {}).get('username', '') + self.databasePassword = self.config.get('database', {}).get('password', '') + self.database = self.config.get('database', {}).get('database', '') + self.readCacheEnabled = self.config.get('database', {}).get('readCacheEnabled', True) + self.cacheReadInterval = int(self.config.get('database', {}).get('cacheReadInterval', 60)) + + if self.databaseType == 'mysql': + self.sqlAlchemyEngine = create_engine(f'mysql://{self.databaseUsername}:{self.databasePassword}@{self.databaseHost}/{self.database}') + self.sqlAlchemySession = sessionmaker(bind=self.sqlAlchemyEngine) + + def sanitizeJson(self, obj): + """ + Handles general JSON sanitizaion. + """ + + if isinstance(obj, datetime.datetime): + return obj.isoformat() + + raise TypeError(f'Object of type {type(obj).__name__} is not JSON serializable') + + def safeClose(self, databaseSession): + try: + if databaseSession.is_active: + databaseSession.close() + except Exception as E: + self.logTool.log(service='Database', level='error', message=f"[Database] [safeClose] Failed to safely close session: {traceback.format_exc()}", redisClient=self.redisLogMessaging) + + async def readDatabase(self): + """ + Reads all database records and caches them into Redis. + """ + while True: + try: + self.logTool.log(service='Database', level='debug', message=f"[Database] [readDatabase] Starting Read from database.", redisClient=self.redisLogMessaging) + databaseMetadata = MetaData() + databaseConnection = self.sqlAlchemyEngine.connect() + databaseMetadata.reflect(bind=databaseConnection) + self.readSession = self.sqlAlchemySession() + + for tableName in databaseMetadata.tables: + tableObject = Table(tableName, databaseMetadata, autoloaded=True) + primaryKeyColumnNames = [primaryKeyColumn.name for primaryKeyColumn in tableObject.primary_key.columns.values()] + if not primaryKeyColumnNames: + continue + primaryKeyName = primaryKeyColumnNames[0] + records = self.readSession.query(tableObject).all() + for record in records: + recordDict = dict(record._mapping) + recordJson = json.dumps(recordDict, default=self.sanitizeJson) + self.logTool.log(service='Database', level='debug', message=f"[Database] [readDatabase] Updating Cache: {recordJson}", redisClient=self.redisLogMessaging) + recordId = primaryKeyName + await(self.redisDatabaseReadMessaging.sendMessage(queue=f'{tableName}', message=f"{recordId}:{recordJson}", queueExpiry=None, usePrefix=True, prefixHostname=self.hostname, prefixServiceName='database')) + try: + self.safeClose(self.readSession) + except: + pass + self.logTool.log(service='Database', level='debug', message=f"[Database] [readDatabase] Finished Read from database.", redisClient=self.redisLogMessaging) + await(asyncio.sleep(self.cacheReadInterval)) + + except Exception as e: + self.logTool.log(service='Database', level='error', message=f"[Database] [readDatabase] Error: {traceback.format_exc()}", redisClient=self.redisLogMessaging) + try: + self.safeClose(self.readSession) + except: + pass + await(asyncio.sleep(self.cacheReadInterval)) + + async def startService(self): + """ + Performs sanity checks on configuration and starts the database service. + """ + await(self.logTool.logAsync(service='Database', level='info', message=f"{self.banners.databaseService()}")) + while True: + + if not self.readCacheEnabled: + await(self.logTool.logAsync(service='Database', level='info', message=f"[Database] [startService] Database read cache enabled, exiting.")) + sys.exit() + + activeTasks = [] + + readCacheTask = asyncio.create_task(self.readDatabase()) + activeTasks.append(readCacheTask) + + completeTasks, pendingTasks = await(asyncio.wait(activeTasks, return_when=asyncio.FIRST_COMPLETED)) + + if len(pendingTasks) > 0: + for pendingTask in pendingTasks: + try: + pendingTask.cancel() + await(asyncio.sleep(0.001)) + except asyncio.CancelledError: + pass + + +if __name__ == '__main__': + databaseService = DatabaseService() + asyncio.run(databaseService.startService()) diff --git a/systemd/pyhss_database.service b/systemd/pyhss_database.service new file mode 100644 index 0000000..9f8d751 --- /dev/null +++ b/systemd/pyhss_database.service @@ -0,0 +1,13 @@ +[Unit] +Description=PyHSS Database Service +PartOf=pyhss.service + + +[Service] +User=root +WorkingDirectory=/etc/pyhss/services/ +ExecStart=python3 databaseService.py +Restart=always + +[Install] +WantedBy=pyhss.service \ No newline at end of file