Browse Source

Add hostnames for blacklist.

Kraeutergarten 6 years ago
parent
commit
b862ce2bfb
2 changed files with 87 additions and 37 deletions
  1. 86 36
      data/Dockerfiles/netfilter/server.py
  2. 1 1
      data/web/inc/functions.fail2ban.inc.php

+ 86 - 36
data/Dockerfiles/netfilter/server.py

@@ -28,8 +28,6 @@ while True:
 
 
 pubsub = r.pubsub()
 pubsub = r.pubsub()
 
 
-resolver = dns.resolver.Resolver()
-
 RULES = {}
 RULES = {}
 RULES[1] = 'warning: .*\[([0-9a-f\.:]+)\]: SASL .+ authentication failed'
 RULES[1] = 'warning: .*\[([0-9a-f\.:]+)\]: SASL .+ authentication failed'
 RULES[2] = '-login: Disconnected \(auth failed, .+\): user=.*, method=.+, rip=([0-9a-f\.:]+),'
 RULES[2] = '-login: Disconnected \(auth failed, .+\): user=.*, method=.+, rip=([0-9a-f\.:]+),'
@@ -40,6 +38,7 @@ RULES[6] = '([0-9a-f\.:]+) \"GET \/SOGo\/.* HTTP.+\" 403 .+'
 #RULES[7] = '-login: Aborted login \(no auth .+\): user=.+, rip=([0-9a-f\.:]+), lip.+'
 #RULES[7] = '-login: Aborted login \(no auth .+\): user=.+, rip=([0-9a-f\.:]+), lip.+'
 
 
 WHITELIST = []
 WHITELIST = []
+BLACKLIST= []
 
 
 bans = {}
 bans = {}
 
 
@@ -83,7 +82,7 @@ def refreshF2boptions():
     try:
     try:
       f2boptions = {}
       f2boptions = {}
       f2boptions = json.loads(r.get('F2B_OPTIONS'))
       f2boptions = json.loads(r.get('F2B_OPTIONS'))
-    except ValueError as e:
+    except ValueError:
       print('Error loading F2B options: F2B_OPTIONS is not json')
       print('Error loading F2B options: F2B_OPTIONS is not json')
       quit_now = True
       quit_now = True
 
 
@@ -132,9 +131,12 @@ def ban(address):
     return
     return
 
 
   self_network = ipaddress.ip_network(address)
   self_network = ipaddress.ip_network(address)
+  
+  with lock:
+    temp_whitelist = set(WHITELIST)
 
 
-  if WHITELIST:
-    for wl_key in WHITELIST:
+  if temp_whitelist:
+    for wl_key in temp_whitelist:
       wl_net = ipaddress.ip_network(wl_key, False)
       wl_net = ipaddress.ip_network(wl_key, False)
           
           
       if wl_net.overlaps(self_network):
       if wl_net.overlaps(self_network):
@@ -210,6 +212,40 @@ def unban(net):
   if net in bans:
   if net in bans:
     del bans[net]
     del bans[net]
 
 
+def permBan(net, unban=False):
+  global lock
+  
+  if type(ipaddress.ip_network(net, strict=False)) is ipaddress.IPv4Network:
+    with lock:
+      chain = iptc.Chain(iptc.Table(iptc.Table.FILTER), 'MAILCOW')
+      rule = iptc.Rule()
+      rule.src = net
+      target = iptc.Target(rule, "REJECT")
+      rule.target = target
+      if rule not in chain.rules and not unban:
+        logCrit('Add host/network %s to blacklist' % net)
+        chain.insert_rule(rule)
+        r.hset('F2B_PERM_BANS', '%s' % net, int(round(time.time()))) 
+      elif rule in chain.rules and unban:
+        logCrit('Remove host/network %s from blacklist' % net)
+        chain.delete_rule(rule)
+        r.hdel('F2B_PERM_BANS', '%s' % net)
+  else:
+    with lock:
+      chain = iptc.Chain(iptc.Table6(iptc.Table6.FILTER), 'MAILCOW')
+      rule = iptc.Rule6()
+      rule.src = net
+      target = iptc.Target(rule, "REJECT")
+      rule.target = target
+      if rule not in chain.rules and not unban:
+        logCrit('Add host/network %s to blacklist' % net)
+        chain.insert_rule(rule)
+        r.hset('F2B_PERM_BANS', '%s' % net, int(round(time.time()))) 
+      elif rule in chain.rules and unban:
+        logCrit('Remove host/network %s from blacklist' % net)
+        chain.delete_rule(rule)
+        r.hdel('F2B_PERM_BANS', '%s' % net)
+    
 def quit(signum, frame):
 def quit(signum, frame):
   global quit_now
   global quit_now
   quit_now = True
   quit_now = True
@@ -353,6 +389,7 @@ def isIpNetwork(address):
 
 
           
           
 def genNetworkList(list):
 def genNetworkList(list):
+  resolver = dns.resolver.Resolver()
   hostnames = []
   hostnames = []
   networks = []
   networks = []
 
 
@@ -366,7 +403,7 @@ def genNetworkList(list):
     hostname_ips = []
     hostname_ips = []
     for rdtype in ['A', 'AAAA']:
     for rdtype in ['A', 'AAAA']:
       try:
       try:
-        answer = resolver.query(qname=hostname, rdtype=rdtype, lifetime=10)
+        answer = resolver.query(qname=hostname, rdtype=rdtype, lifetime=3)
       except dns.exception.Timeout:
       except dns.exception.Timeout:
         logInfo('Hostname %s timedout on resolve' % hostname)
         logInfo('Hostname %s timedout on resolve' % hostname)
         break
         break
@@ -381,7 +418,7 @@ def genNetworkList(list):
 
 
     networks.extend(hostname_ips)
     networks.extend(hostname_ips)
       
       
-  return networks
+  return set(networks)
 
 
 def whitelistUpdate():
 def whitelistUpdate():
   global lock
   global lock
@@ -392,16 +429,48 @@ def whitelistUpdate():
     start_time = time.time()
     start_time = time.time()
     list = r.hgetall('F2B_WHITELIST')
     list = r.hgetall('F2B_WHITELIST')
     
     
-    gen_whitelist = []
+    new_whitelist = []
     
     
     if list:
     if list:
-      gen_whitelist = genNetworkList(list)
+      new_whitelist = genNetworkList(list)
     
     
-    if Counter(gen_whitelist) != Counter(WHITELIST):
-      WHITELIST = gen_whitelist
-      logInfo('New entrys for whitelist %s' % WHITELIST)
+    with lock:
+      if Counter(new_whitelist) != Counter(WHITELIST):
+        WHITELIST = new_whitelist
+        logInfo('Whitelist was changed, it has %s entries' % len(WHITELIST))
 
 
     time.sleep(60.0 - ((time.time() - start_time) % 60.0)) 
     time.sleep(60.0 - ((time.time() - start_time) % 60.0)) 
+    
+def blacklistUpdate():
+  global quit_now
+  global BLACKLIST
+  
+  while not quit_now:
+    start_time = time.time()
+    list = r.hgetall('F2B_BLACKLIST')
+    
+    new_blacklist = []
+    
+    if list:
+      new_blacklist = genNetworkList(list)
+      
+    if Counter(new_blacklist) != Counter(BLACKLIST): 
+      addban = set(new_blacklist).difference(BLACKLIST)
+      delban = set(BLACKLIST).difference(new_blacklist)
+        
+      BLACKLIST = new_blacklist
+      logInfo('Blacklist was changed, it has %s entries' % len(BLACKLIST))
+        
+      if addban:
+        for net in addban:
+          permBan(net=net)
+            
+      if delban:
+        for net in delban:
+          permBan(net=net, unban=True)
+      
+        
+    time.sleep(60.0 - ((time.time() - start_time) % 60.0)) 
       
       
 def initChain():
 def initChain():
   # Is called before threads start, no locking
   # Is called before threads start, no locking
@@ -430,30 +499,7 @@ def initChain():
     rule.target = target
     rule.target = target
     if rule not in chain.rules:
     if rule not in chain.rules:
       chain.insert_rule(rule)
       chain.insert_rule(rule)
-  # Apply blacklist
-  BLACKLIST = r.hgetall('F2B_BLACKLIST')
-  if BLACKLIST:
-    for bl_key in BLACKLIST:
-      if type(ipaddress.ip_network(bl_key, strict=False)) is ipaddress.IPv4Network:
-        chain = iptc.Chain(iptc.Table(iptc.Table.FILTER), 'MAILCOW')
-        rule = iptc.Rule()
-        rule.src = bl_key
-        target = iptc.Target(rule, "REJECT")
-        rule.target = target
-        if rule not in chain.rules:
-          logCrit('Blacklisting host/network %s' % bl_key)
-          chain.insert_rule(rule)
-          r.hset('F2B_PERM_BANS', '%s' % bl_key, int(round(time.time())))
-      else:
-        chain = iptc.Chain(iptc.Table6(iptc.Table6.FILTER), 'MAILCOW')
-        rule = iptc.Rule6()
-        rule.src = bl_key
-        target = iptc.Target(rule, "REJECT")
-        rule.target = target
-        if rule not in chain.rules:
-          logCrit('Blacklisting host/network %s' % bl_key)
-          chain.insert_rule(rule)
-          r.hset('F2B_PERM_BANS', '%s' % bl_key, int(round(time.time())))
+ 
 
 
 if __name__ == '__main__':
 if __name__ == '__main__':
 
 
@@ -495,6 +541,10 @@ if __name__ == '__main__':
   mailcowchainwatch_thread = Thread(target=mailcowChainOrder)
   mailcowchainwatch_thread = Thread(target=mailcowChainOrder)
   mailcowchainwatch_thread.daemon = True
   mailcowchainwatch_thread.daemon = True
   mailcowchainwatch_thread.start()
   mailcowchainwatch_thread.start()
+  
+  blacklistupdate_thread = Thread(target=blacklistUpdate)
+  blacklistupdate_thread.daemon = True
+  blacklistupdate_thread.start()
 
 
   whitelistupdate_thread = Thread(target=whitelistUpdate)
   whitelistupdate_thread = Thread(target=whitelistUpdate)
   whitelistupdate_thread.daemon = True
   whitelistupdate_thread.daemon = True

+ 1 - 1
data/web/inc/functions.fail2ban.inc.php

@@ -203,7 +203,7 @@ function fail2ban($_action, $_data = null) {
           $bl_array = array_map('trim', preg_split( "/( |,|;|\n)/", $bl));
           $bl_array = array_map('trim', preg_split( "/( |,|;|\n)/", $bl));
           if (is_array($bl_array)) {
           if (is_array($bl_array)) {
             foreach ($bl_array as $bl_item) {
             foreach ($bl_array as $bl_item) {
-              if (valid_network($bl_item)) {
+              if (valid_network($bl_item) || valid_hostname($bl_item)) {
                 $redis->hSet('F2B_BLACKLIST', $bl_item, 1);
                 $redis->hSet('F2B_BLACKLIST', $bl_item, 1);
               }
               }
             }
             }