diff --git a/ZenPacks/zenoss/PostgreSQL/Table.py b/ZenPacks/zenoss/PostgreSQL/Table.py
index 4733463..345ff30 100644
--- a/ZenPacks/zenoss/PostgreSQL/Table.py
+++ b/ZenPacks/zenoss/PostgreSQL/Table.py
@@ -60,5 +60,7 @@ class Table(DeviceComponent, ManagedEntity, CollectedOrModeledMixin):
event_key = "ComponentId"
def device(self):
- return self.database().device()
-
+ db = self.database()
+ if db:
+ return db.device()
+ return None
diff --git a/ZenPacks/zenoss/PostgreSQL/__init__.py b/ZenPacks/zenoss/PostgreSQL/__init__.py
index 5963237..6efd97e 100644
--- a/ZenPacks/zenoss/PostgreSQL/__init__.py
+++ b/ZenPacks/zenoss/PostgreSQL/__init__.py
@@ -12,6 +12,7 @@
###########################################################################
import logging
+
log = logging.getLogger('zen.PostgreSQL')
import os
@@ -22,6 +23,7 @@
from Products.ZenRelations.RelSchema import ToManyCont, ToOne
from Products.ZenUtils.Utils import monkeypatch, zenPath
+
class ZenPack(ZenPackBase):
packZProperties = [
('zPostgreSQLPort', 5432, 'int'),
@@ -29,8 +31,16 @@ class ZenPack(ZenPackBase):
('zPostgreSQLPassword', '', 'password'),
('zPostgreSQLUseSSL', False, 'boolean'),
('zPostgreSQLDefaultDB', 'postgres', 'string'),
+ ('zPostgreSQLTableRegex', [], 'lines'),
]
+ packZProperties_data = {
+ 'zPostgreSQLTableRegex': {
+ 'description': "List of regular expressions (matched against table names) to control which tables are NOT modeled from All databases.",
+ 'label': "Regex Table Filter",
+ 'type': "lines" },
+ }
+
def install(self, app):
super(ZenPack, self).install(app)
self.patchPostgreSQLDriver()
@@ -40,10 +50,10 @@ def remove(self, app, leaveObjects=False):
if not leaveObjects:
# Remove our custom relations addition.
Device._relations = tuple(
- [ x for x in Device._relations if x[0] != 'pgDatabases' ])
+ [x for x in Device._relations if x[0] != 'pgDatabases'])
self.updateExistingRelations(app.zport.dmd)
-
+
# Revert all pg8000 library patches
self.patchPostgreSQLDriver(revert=True)
@@ -57,21 +67,22 @@ def updateExistingRelations(self, dmd):
def patchPostgreSQLDriver(self, revert=False):
log.info('Patching pg8000 core library')
patch_dir = self.path('lib')
-
+
# Getting a list of all patches which will be applied
patches_list = [file for file in os.listdir(patch_dir) if file.endswith('.patch')]
-
- cmd = "patch -p0 -d %s -i %s" if not revert else "patch -p0 -R -d %s -i %s"
+
+ cmd = "patch -p0 -d %s -i %s" if not revert else "patch -p0 -R -d %s -i %s"
for patch in patches_list:
os.system(cmd % (
os.path.join(patch_dir, 'pg8000'),
os.path.join(patch_dir, patch)
))
+
# Allow PostgreSQL databases to be related to any device.
Device._relations += (
('pgDatabases', ToManyCont(ToOne,
- 'ZenPacks.zenoss.PostgreSQL.Database.Database', 'server')),
+ 'ZenPacks.zenoss.PostgreSQL.Database.Database', 'server')),
)
# We need to filter components by id instead of name.
@@ -79,6 +90,7 @@ def patchPostgreSQLDriver(self, revert=False):
"\"(device = '%s' and component = '%s')\""
" % (me.device().getDmdKey(), me.id)")
+
@monkeypatch('Products.ZenModel.Device.Device')
def setPostgreSQL(self, active=True):
if not active:
@@ -96,11 +108,12 @@ def setPostgreSQL(self, active=True):
if device.zCommandCommandTimeout < 180:
device.setZenProperty('zCommandCommandTimeout', 180)
+
@monkeypatch('Products.ZenModel.Device.Device')
def getPostgreSQL(self):
device = self.primaryAq()
if 'PostgreSQLServer' in device.zDeviceTemplates \
- and device.zCommandCommandTimeout >= 180:
+ and device.zCommandCommandTimeout >= 180:
return True
return False
diff --git a/ZenPacks/zenoss/PostgreSQL/modeler/plugins/zenoss/PostgreSQL.py b/ZenPacks/zenoss/PostgreSQL/modeler/plugins/zenoss/PostgreSQL.py
index 9649e74..66f7c7c 100644
--- a/ZenPacks/zenoss/PostgreSQL/modeler/plugins/zenoss/PostgreSQL.py
+++ b/ZenPacks/zenoss/PostgreSQL/modeler/plugins/zenoss/PostgreSQL.py
@@ -12,13 +12,16 @@
###########################################################################
import logging
+import re
+
log = logging.getLogger('zen.PostgreSQL')
from Products.DataCollector.plugins.CollectorPlugin import PythonPlugin
from Products.DataCollector.plugins.DataMaps import ObjectMap, RelationshipMap
from Products.ZenUtils.Utils import prepId
-from ZenPacks.zenoss.PostgreSQL.util import PgHelper
+from ZenPacks.zenoss.PostgreSQL.util import PgHelper, exclude_patterns_list, is_suppressed
+
class PostgreSQL(PythonPlugin):
deviceProperties = PythonPlugin.deviceProperties + (
@@ -27,6 +30,7 @@ class PostgreSQL(PythonPlugin):
'zPostgreSQLPassword',
'zPostgreSQLUseSSL',
'zPostgreSQLDefaultDB',
+ 'zPostgreSQLTableRegex',
)
def collect(self, device, unused):
@@ -39,6 +43,7 @@ def collect(self, device, unused):
device.zPostgreSQLDefaultDB)
results = {}
+ exclude_patterns = exclude_patterns_list(getattr(device, 'zPostgreSQLTableRegex', []))
log.info("Getting database list")
try:
@@ -55,8 +60,14 @@ def collect(self, device, unused):
log.info("Getting tables list for {0}".format(dbName))
try:
- results['databases'][dbName]['tables'] = \
- pg.getTablesInDatabase(dbName)
+ tables = pg.getTablesInDatabase(dbName)
+ if exclude_patterns:
+ for key in tables.keys():
+ if is_suppressed(key, exclude_patterns):
+ del tables[key]
+
+ results['databases'][dbName]['tables'] = tables
+
except Exception, ex:
log.warn("Error getting tables list for {0}: {1}".format(
dbName, ex))
@@ -69,7 +80,7 @@ def process(self, devices, results, unused):
if results is None:
return None
- maps = [ self.objectMap(dict(setPostgreSQL=True)) ]
+ maps = [self.objectMap(dict(setPostgreSQL=True))]
databases = []
for dbName, dbDetail in results['databases'].items():
diff --git a/ZenPacks/zenoss/PostgreSQL/util.py b/ZenPacks/zenoss/PostgreSQL/util.py
index d701d2e..8a5823c 100644
--- a/ZenPacks/zenoss/PostgreSQL/util.py
+++ b/ZenPacks/zenoss/PostgreSQL/util.py
@@ -15,7 +15,9 @@
import math
import sys
import time
-
+import re
+import logging
+LOG = logging.getLogger('zen.PostgreSQL.utils')
def addLocalLibPath():
"""
@@ -545,3 +547,28 @@ def getTableStatsForDatabase(self, db):
cursor.close()
return tableStats
+
+def exclude_patterns_list(excludes):
+ exclude_patterns = []
+
+ for exclude in excludes:
+ exclude = exclude.strip()
+ if exclude == "" or exclude.startswith("#"):
+ continue
+
+ try:
+ exclude_patterns.append(re.compile(exclude))
+ except Exception:
+ LOG.warn("Invalid zPostgreSQLTableRegex value: '%s', this modeling filter will not be applied.", exclude)
+ continue
+
+ return exclude_patterns
+
+
+def is_suppressed(item, exclude_patterns):
+
+ for exclude_pattern in exclude_patterns:
+ if exclude_pattern.search(item):
+ return True
+
+ return False
diff --git a/docs/body.md b/docs/body.md
index 2e54c86..84ffdba 100644
--- a/docs/body.md
+++ b/docs/body.md
@@ -82,6 +82,7 @@ individual devices.
- *zPostgreSQLUsername* - Must be a superuser. Default: postgres
- *zPostgreSQLPassword* - Password for user. No default.
- *zPostgreSQLDefaultDB* - Default database. Default: postgres
+ - *zPostgreSQLTableRegex* - Filter tables of all databases with Regex. Default: ""
In addition to setting these properties you must add the ''zenoss.PostgreSQL''
modeler plugin to a device class or individual device. This modeler plugin will
@@ -164,6 +165,11 @@ zSnmpMonitorIgnore property to True and remodel.
Changes
---------------
+1.0.13
+
+* Make PostgreSQL table modeling optional (ZPS-8554)
+* Tested with Zenoss Cloud, Zenoss 6.7.0 and Service Impact 5.6.1
+
1.0.12
* Resolved isuue with error in pg8000 library on Ubuntu OS (ZPS-7424)
diff --git a/docs/releases.md b/docs/releases.md
index e9a3fce..03599f0 100644
--- a/docs/releases.md
+++ b/docs/releases.md
@@ -1,6 +1,10 @@
Releases
--------
+Version 1.0.13-Download
+Released on 2023/9/12
+Compatible with Zenoss 6.x, Zenoss Cloud and Service Impact 5.6.1
+
Version 1.0.12-Download
Released on 2021/1/14
Compatible with Zenoss 6.4.1 - 6.5.0, Zenoss Cloud and Service Impact 5.5.3
diff --git a/setup.py b/setup.py
index becb7f4..56dab97 100644
--- a/setup.py
+++ b/setup.py
@@ -16,7 +16,7 @@
# or saved. Do not modify them directly here.
# NB: PACKAGES is deprecated
NAME = "ZenPacks.zenoss.PostgreSQL"
-VERSION = "1.0.12"
+VERSION = "1.0.13"
AUTHOR = "Zenoss"
LICENSE = "GPLv2"
NAMESPACE_PACKAGES = ['ZenPacks', 'ZenPacks.zenoss']