aboutsummaryrefslogtreecommitdiff
path: root/modules/dnsbl
diff options
context:
space:
mode:
authorGravatar jesopo2019-12-12 14:13:17 +0000
committerGravatar jesopo2019-12-12 14:13:17 +0000
commitfbe1acf220b69e244776105390b1abf5b77a96e8 (patch)
tree46dfcb8182ea2a069809b936e5ae72271995ad89 /modules/dnsbl
parentlist.insert() takes an index! (diff)
refactor dnsbl module, show reason for positive detection when possible
Diffstat (limited to 'modules/dnsbl')
-rw-r--r--modules/dnsbl/__init__.py53
-rw-r--r--modules/dnsbl/lists.py40
2 files changed, 93 insertions, 0 deletions
diff --git a/modules/dnsbl/__init__.py b/modules/dnsbl/__init__.py
new file mode 100644
index 00000000..2c3fa36d
--- /dev/null
+++ b/modules/dnsbl/__init__.py
@@ -0,0 +1,53 @@
+import ipaddress
+from src import ModuleManager, utils
+import dns.resolver
+from . import lists as _lists
+
+class Module(ModuleManager.BaseModule):
+ @utils.hook("received.command.dnsbl")
+ def dnsbl(self, event):
+ args = event["args_split"]
+
+ default_lists = _lists.default_lists()
+ lists = []
+ for i, arg in reversed(list(enumerate(args))):
+ if arg[0] == "@":
+ hostname = args.pop(i)
+ if hostname in default_lists:
+ lists.insert(0, default_lists[hostname])
+ else:
+ lists.insert(0, lists.DNSBL(hostname))
+
+ lists = lists or list(default_lists.values())
+
+ address = args[0]
+ failed = self._check_lists(lists, address)
+ if failed:
+ failed = ["%s (%s)" % item for item in failed]
+ event["stderr"].write("%s failed for lists: %s" %
+ (address, ", ".join(failed)))
+ else:
+ event["stdout"].write("%s not found in blacklists" % address)
+
+ def _check_lists(self, lists, address):
+ address_obj = ipaddress.ip_address(address)
+
+ if address_obj.version == 6:
+ address = reversed(address_obj.exploded.replace(":", ""))
+ else:
+ address = reversed(address.split("."))
+ address = ".".join(address)
+
+ failed = []
+ for list in lists:
+ record = self._check_list(list.hostname, address)
+ if not record == None:
+ failed.append((list.hostname, list.process(record)))
+ return failed
+
+ def _check_list(self, list, address):
+ list_address = "%s.%s" % (address, list)
+ try:
+ return dns.resolver.query(list_address, "A")[0].to_text()
+ except dns.resolver.NXDOMAIN:
+ return None
diff --git a/modules/dnsbl/lists.py b/modules/dnsbl/lists.py
new file mode 100644
index 00000000..0ae6a1b6
--- /dev/null
+++ b/modules/dnsbl/lists.py
@@ -0,0 +1,40 @@
+import collections
+
+class DNSBL(object):
+ def __init__(self, hostname=None):
+ if not hostname == None:
+ self.hostname = hostname
+
+ def process(self, result: str):
+ return "unknown"
+
+class ZenSpamhaus(DNSBL):
+ hostname = "zen.spamhaus.org"
+ def process(self, result):
+ result = result.rsplit(".", 1)[1]
+ if result in ["2", "3", "9"]:
+ return "spam"
+ elif result in ["4", "5", "6", "7"]:
+ return "exploits"
+class EFNetRBL(DNSBL):
+ hostname = "rbl.efnetrbl.org"
+ SPAMTRAP = ["2", "3"]
+ def process(self, result):
+ result = result.rsplit(".", 1)[1]
+ if result == "1":
+ return "proxy"
+ elif result in self.SPAMTRAP:
+ return "spamtap"
+ elif result == "4":
+ return "tor"
+ elif result == "5":
+ return "flooding"
+
+DEFAULT_LISTS = [
+ ZenSpamhaus(),
+ EFNetRBL()
+]
+
+def default_lists():
+ return collections.OrderedDict(
+ (dnsbl.hostname, dnsbl) for dnsbl in DEFAULT_LISTS)