Arthur de Jong

Open Source / Free Software developer

summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorArthur de Jong <arthur@arthurdejong.org>2013-03-29 20:14:49 +0100
committerArthur de Jong <arthur@arthurdejong.org>2013-03-29 20:17:04 +0100
commita75cfb9d5f67dd4be325f151f9e0fc9af0864ac2 (patch)
treec99a498301ee2899150ced655437dd61844c0454
parent585d38894b32551fba68ea9b202c12ce5f8cc966 (diff)
Detect and handle connection failure and recovery
Logs a connection recovery message and run a nscd cache invalidation if configured.
-rw-r--r--pynslcd/nscd.py7
-rw-r--r--pynslcd/search.py22
2 files changed, 28 insertions, 1 deletions
diff --git a/pynslcd/nscd.py b/pynslcd/nscd.py
index 89cb483..19e5ceb 100644
--- a/pynslcd/nscd.py
+++ b/pynslcd/nscd.py
@@ -24,6 +24,8 @@ import os
import subprocess
import struct
+import cfg
+
# the file descriptor used for sending messages to the child process
signalfd = None
@@ -100,7 +102,10 @@ def start_invalidator():
def invalidate(db=None):
if signalfd is None:
return # nothing to do
- db = _db_to_char.get(db, '')
+ if db:
+ db = _db_to_char.get(db, '')
+ else:
+ db = ''.join(_db_to_char[x] for x in cfg.nscd_invalidate)
try:
os.write(signalfd, db)
except:
diff --git a/pynslcd/search.py b/pynslcd/search.py
index 9629bec..219929b 100644
--- a/pynslcd/search.py
+++ b/pynslcd/search.py
@@ -25,6 +25,11 @@ import ldap
import ldap.ldapobject
import cfg
+import nscd
+
+
+# global indicator that there was some error connection to an LDAP server
+server_error = False
class Connection(ldap.ldapobject.ReconnectLDAPObject):
@@ -50,6 +55,23 @@ class Connection(ldap.ldapobject.ReconnectLDAPObject):
if cfg.ssl or cfg.uri.startswith('ldaps://'):
self.set_option(ldap.OPT_X_TLS, ldap.OPT_X_TLS_HARD)
+ def reconnect_after_fail(self):
+ logging.info('connected to LDAP server %s', cfg.uri)
+ nscd.invalidate()
+
+ def search_s(self, *args, **kwargs):
+ # wrapper function to keep the global server_error state
+ global server_error
+ try:
+ res = ldap.ldapobject.ReconnectLDAPObject.search_s(self, *args, **kwargs)
+ except ldap.SERVER_DOWN:
+ server_error = True
+ raise
+ if server_error:
+ self.reconnect_after_fail()
+ server_error = False
+ return res
+
class LDAPSearch(object):
"""