Source code for keycloak.connection

# -*- coding: utf-8 -*-
#
# The MIT License (MIT)
#
# Copyright (C) 2017 Marcos Pereira <marcospereira.mpj@gmail.com>
#
# Permission is hereby granted, free of charge, to any person obtaining a copy of
# this software and associated documentation files (the "Software"), to deal in
# the Software without restriction, including without limitation the rights to
# use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
# the Software, and to permit persons to whom the Software is furnished to do so,
# subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
# FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
# COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
# IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

"""Connection manager module."""

try:
    from urllib.parse import urljoin
except ImportError:  # pragma: no cover
    from urlparse import urljoin

import requests
from requests.adapters import HTTPAdapter

from .exceptions import KeycloakConnectionError


[docs]class ConnectionManager(object): """Represents a simple server connection. :param base_url: The server URL. :type base_url: str :param headers: The header parameters of the requests to the server. :type headers: dict :param timeout: Timeout to use for requests to the server. :type timeout: int :param verify: Boolean value to enable or disable certificate validation or a string containing a path to a CA bundle to use :type verify: Union[bool,str] :param proxies: The proxies servers requests is sent by. :type proxies: dict """ def __init__(self, base_url, headers={}, timeout=60, verify=True, proxies=None): """Init method. :param base_url: The server URL. :type base_url: str :param headers: The header parameters of the requests to the server. :type headers: dict :param timeout: Timeout to use for requests to the server. :type timeout: int :param verify: Boolean value to enable or disable certificate validation or a string containing a path to a CA bundle to use :type verify: Union[bool,str] :param proxies: The proxies servers requests is sent by. :type proxies: dict """ self.base_url = base_url self.headers = headers self.timeout = timeout self.verify = verify self._s = requests.Session() self._s.auth = lambda x: x # don't let requests add auth headers # retry once to reset connection with Keycloak after tomcat's ConnectionTimeout # see https://github.com/marcospereirampj/python-keycloak/issues/36 for protocol in ("https://", "http://"): adapter = HTTPAdapter(max_retries=1) # adds POST to retry whitelist allowed_methods = set(adapter.max_retries.allowed_methods) allowed_methods.add("POST") adapter.max_retries.allowed_methods = frozenset(allowed_methods) self._s.mount(protocol, adapter) if proxies: self._s.proxies.update(proxies)
[docs] def __del__(self): """Del method.""" if hasattr(self, "_s"): self._s.close()
@property
[docs] def base_url(self): """Return base url in use for requests to the server. :returns: Base URL :rtype: str """ return self._base_url
@base_url.setter def base_url(self, value): self._base_url = value @property
[docs] def timeout(self): """Return timeout in use for request to the server. :returns: Timeout :rtype: int """ return self._timeout
@timeout.setter def timeout(self, value): self._timeout = value @property
[docs] def verify(self): """Return verify in use for request to the server. :returns: Verify indicator :rtype: bool """ return self._verify
@verify.setter def verify(self, value): self._verify = value @property
[docs] def headers(self): """Return header request to the server. :returns: Request headers :rtype: dict """ return self._headers
@headers.setter def headers(self, value): self._headers = value
[docs] def param_headers(self, key): """Return a specific header parameter. :param key: Header parameters key. :type key: str :returns: If the header parameters exist, return its value. :rtype: str """ return self.headers.get(key)
[docs] def clean_headers(self): """Clear header parameters.""" self.headers = {}
[docs] def exist_param_headers(self, key): """Check if the parameter exists in the header. :param key: Header parameters key. :type key: str :returns: If the header parameters exist, return True. :rtype: bool """ return self.param_headers(key) is not None
[docs] def add_param_headers(self, key, value): """Add a single parameter inside the header. :param key: Header parameters key. :type key: str :param value: Value to be added. :type value: str """ self.headers[key] = value
[docs] def del_param_headers(self, key): """Remove a specific parameter. :param key: Key of the header parameters. :type key: str """ self.headers.pop(key, None)
[docs] def raw_get(self, path, **kwargs): """Submit get request to the path. :param path: Path for request. :type path: str :param kwargs: Additional arguments :type kwargs: dict :returns: Response the request. :rtype: Response :raises KeycloakConnectionError: HttpError Can't connect to server. """ try: return self._s.get( urljoin(self.base_url, path), params=kwargs, headers=self.headers, timeout=self.timeout, verify=self.verify, ) except Exception as e: raise KeycloakConnectionError("Can't connect to server (%s)" % e)
[docs] def raw_post(self, path, data, **kwargs): """Submit post request to the path. :param path: Path for request. :type path: str :param data: Payload for request. :type data: dict :param kwargs: Additional arguments :type kwargs: dict :returns: Response the request. :rtype: Response :raises KeycloakConnectionError: HttpError Can't connect to server. """ try: return self._s.post( urljoin(self.base_url, path), params=kwargs, data=data, headers=self.headers, timeout=self.timeout, verify=self.verify, ) except Exception as e: raise KeycloakConnectionError("Can't connect to server (%s)" % e)
[docs] def raw_put(self, path, data, **kwargs): """Submit put request to the path. :param path: Path for request. :type path: str :param data: Payload for request. :type data: dict :param kwargs: Additional arguments :type kwargs: dict :returns: Response the request. :rtype: Response :raises KeycloakConnectionError: HttpError Can't connect to server. """ try: return self._s.put( urljoin(self.base_url, path), params=kwargs, data=data, headers=self.headers, timeout=self.timeout, verify=self.verify, ) except Exception as e: raise KeycloakConnectionError("Can't connect to server (%s)" % e)
[docs] def raw_delete(self, path, data=None, **kwargs): """Submit delete request to the path. :param path: Path for request. :type path: str :param data: Payload for request. :type data: dict | None :param kwargs: Additional arguments :type kwargs: dict :returns: Response the request. :rtype: Response :raises KeycloakConnectionError: HttpError Can't connect to server. """ try: return self._s.delete( urljoin(self.base_url, path), params=kwargs, data=data or dict(), headers=self.headers, timeout=self.timeout, verify=self.verify, ) except Exception as e: raise KeycloakConnectionError("Can't connect to server (%s)" % e)