|
|
483b06 |
From 0367e6dd66d47a78484e12e76119d9662356bf48 Mon Sep 17 00:00:00 2001
|
|
|
483b06 |
From: Stanislav Laznicka <slaznick@redhat.com>
|
|
|
483b06 |
Date: Fri, 26 May 2017 08:37:36 +0200
|
|
|
483b06 |
Subject: [PATCH] Avoid possible endless recursion in RPC call
|
|
|
483b06 |
|
|
|
483b06 |
This commit removes recursion in RPCClient.forward() which may lack
|
|
|
483b06 |
end condition.
|
|
|
483b06 |
|
|
|
483b06 |
https://pagure.io/freeipa/issue/6796
|
|
|
483b06 |
|
|
|
483b06 |
Reviewed-By: Florence Blanc-Renaud <frenaud@redhat.com>
|
|
|
483b06 |
---
|
|
|
483b06 |
ipalib/rpc.py | 95 +++++++++++++++++++++++++++++++++--------------------------
|
|
|
483b06 |
1 file changed, 54 insertions(+), 41 deletions(-)
|
|
|
483b06 |
|
|
|
483b06 |
diff --git a/ipalib/rpc.py b/ipalib/rpc.py
|
|
|
483b06 |
index e23ca3d061645b2695a9e0deaa0b7d666f986e0e..297ed80414fae3d8b27558567425fec704f3e862 100644
|
|
|
483b06 |
--- a/ipalib/rpc.py
|
|
|
483b06 |
+++ b/ipalib/rpc.py
|
|
|
483b06 |
@@ -1088,50 +1088,63 @@ class RPCClient(Connectible):
|
|
|
483b06 |
:param kw: Keyword arguments to pass to remote command.
|
|
|
483b06 |
"""
|
|
|
483b06 |
server = getattr(context, 'request_url', None)
|
|
|
483b06 |
- self.log.info("Forwarding '%s' to %s server '%s'",
|
|
|
483b06 |
- name, self.protocol, server)
|
|
|
483b06 |
command = getattr(self.conn, name)
|
|
|
483b06 |
params = [args, kw]
|
|
|
483b06 |
- try:
|
|
|
483b06 |
- return self._call_command(command, params)
|
|
|
483b06 |
- except Fault as e:
|
|
|
483b06 |
- e = decode_fault(e)
|
|
|
483b06 |
- self.debug('Caught fault %d from server %s: %s', e.faultCode,
|
|
|
483b06 |
- server, e.faultString)
|
|
|
483b06 |
- if e.faultCode in errors_by_code:
|
|
|
483b06 |
- error = errors_by_code[e.faultCode]
|
|
|
483b06 |
- raise error(message=e.faultString)
|
|
|
483b06 |
- raise UnknownError(
|
|
|
483b06 |
- code=e.faultCode,
|
|
|
483b06 |
- error=e.faultString,
|
|
|
483b06 |
- server=server,
|
|
|
483b06 |
- )
|
|
|
483b06 |
- except SSLError as e:
|
|
|
483b06 |
- raise NetworkError(uri=server, error=str(e))
|
|
|
483b06 |
- except ProtocolError as e:
|
|
|
483b06 |
- # By catching a 401 here we can detect the case where we have
|
|
|
483b06 |
- # a single IPA server and the session is invalid. Otherwise
|
|
|
483b06 |
- # we always have to do a ping().
|
|
|
483b06 |
- session_cookie = getattr(context, 'session_cookie', None)
|
|
|
483b06 |
- if session_cookie and e.errcode == 401:
|
|
|
483b06 |
- # Unauthorized. Remove the session and try again.
|
|
|
483b06 |
- delattr(context, 'session_cookie')
|
|
|
483b06 |
- try:
|
|
|
483b06 |
- principal = getattr(context, 'principal', None)
|
|
|
483b06 |
- delete_persistent_client_session_data(principal)
|
|
|
483b06 |
- except Exception as e:
|
|
|
483b06 |
- # This shouldn't happen if we have a session but it isn't fatal.
|
|
|
483b06 |
- pass
|
|
|
483b06 |
|
|
|
483b06 |
- # Create a new serverproxy with the non-session URI
|
|
|
483b06 |
- serverproxy = self.create_connection(os.environ.get('KRB5CCNAME'), self.env.verbose, self.env.fallback, self.env.delegate)
|
|
|
483b06 |
- setattr(context, self.id, Connection(serverproxy, self.disconnect))
|
|
|
483b06 |
- return self.forward(name, *args, **kw)
|
|
|
483b06 |
- raise NetworkError(uri=server, error=e.errmsg)
|
|
|
483b06 |
- except socket.error as e:
|
|
|
483b06 |
- raise NetworkError(uri=server, error=str(e))
|
|
|
483b06 |
- except (OverflowError, TypeError) as e:
|
|
|
483b06 |
- raise XMLRPCMarshallError(error=str(e))
|
|
|
483b06 |
+ # we'll be trying to connect multiple times with a new session cookie
|
|
|
483b06 |
+ # each time should we be getting UNAUTHORIZED error from the server
|
|
|
483b06 |
+ max_tries = 5
|
|
|
483b06 |
+ for try_num in range(0, max_tries):
|
|
|
483b06 |
+ self.log.info("[try %d]: Forwarding '%s' to %s server '%s'",
|
|
|
483b06 |
+ try_num+1, name, self.protocol, server)
|
|
|
483b06 |
+ try:
|
|
|
483b06 |
+ return self._call_command(command, params)
|
|
|
483b06 |
+ except Fault as e:
|
|
|
483b06 |
+ e = decode_fault(e)
|
|
|
483b06 |
+ self.debug('Caught fault %d from server %s: %s', e.faultCode,
|
|
|
483b06 |
+ server, e.faultString)
|
|
|
483b06 |
+ if e.faultCode in errors_by_code:
|
|
|
483b06 |
+ error = errors_by_code[e.faultCode]
|
|
|
483b06 |
+ raise error(message=e.faultString)
|
|
|
483b06 |
+ raise UnknownError(
|
|
|
483b06 |
+ code=e.faultCode,
|
|
|
483b06 |
+ error=e.faultString,
|
|
|
483b06 |
+ server=server,
|
|
|
483b06 |
+ )
|
|
|
483b06 |
+ except ProtocolError as e:
|
|
|
483b06 |
+ # By catching a 401 here we can detect the case where we have
|
|
|
483b06 |
+ # a single IPA server and the session is invalid. Otherwise
|
|
|
483b06 |
+ # we always have to do a ping().
|
|
|
483b06 |
+ session_cookie = getattr(context, 'session_cookie', None)
|
|
|
483b06 |
+ if session_cookie and e.errcode == 401:
|
|
|
483b06 |
+ # Unauthorized. Remove the session and try again.
|
|
|
483b06 |
+ delattr(context, 'session_cookie')
|
|
|
483b06 |
+ try:
|
|
|
483b06 |
+ principal = getattr(context, 'principal', None)
|
|
|
483b06 |
+ delete_persistent_client_session_data(principal)
|
|
|
483b06 |
+ except Exception as e:
|
|
|
483b06 |
+ # This shouldn't happen if we have a session
|
|
|
483b06 |
+ # but it isn't fatal.
|
|
|
483b06 |
+ self.debug("Error trying to remove persisent session "
|
|
|
483b06 |
+ "data: {err}".format(err=e))
|
|
|
483b06 |
+
|
|
|
483b06 |
+ # Create a new serverproxy with the non-session URI
|
|
|
483b06 |
+ serverproxy = self.create_connection(
|
|
|
483b06 |
+ os.environ.get('KRB5CCNAME'), self.env.verbose,
|
|
|
483b06 |
+ self.env.fallback, self.env.delegate)
|
|
|
483b06 |
+
|
|
|
483b06 |
+ setattr(context, self.id,
|
|
|
483b06 |
+ Connection(serverproxy, self.disconnect))
|
|
|
483b06 |
+ # try to connect again with the new session cookie
|
|
|
483b06 |
+ continue
|
|
|
483b06 |
+ raise NetworkError(uri=server, error=e.errmsg)
|
|
|
483b06 |
+ except (SSLError, socket.error) as e:
|
|
|
483b06 |
+ raise NetworkError(uri=server, error=str(e))
|
|
|
483b06 |
+ except (OverflowError, TypeError) as e:
|
|
|
483b06 |
+ raise XMLRPCMarshallError(error=str(e))
|
|
|
483b06 |
+ raise NetworkError(
|
|
|
483b06 |
+ uri=server,
|
|
|
483b06 |
+ error=_("Exceeded number of tries to forward a request."))
|
|
|
483b06 |
|
|
|
483b06 |
|
|
|
483b06 |
class xmlclient(RPCClient):
|
|
|
483b06 |
--
|
|
|
483b06 |
2.9.4
|
|
|
483b06 |
|