Source code for warthog.transport

# -*- coding: utf-8 -*-
#
# Warthog - Simple client for A10 load balancers
#
# Copyright 2014-2016 Smarter Travel
#
# Available under the MIT license. See LICENSE for details.
#

"""
warthog.transport
~~~~~~~~~~~~~~~~~

Methods to configure how to interact with the load balancer API over HTTP or HTTPS.
"""

import warnings

import requests

from requests.adapters import (
    HTTPAdapter,
    DEFAULT_POOLBLOCK,
    DEFAULT_POOLSIZE)
from urllib3.exceptions import InsecureRequestWarning
from urllib3.poolmanager import PoolManager

import warthog.ssl

# Default to using the SSL/TLS version that the A10 requires instead of
# the default that the requests/urllib3 library picks. Or, maybe the A10
# just doesn't allow the client to negotiate. Either way, we use TLSv1.2.
DEFAULT_SSL_VERSION = warthog.ssl.PROTOCOL_TLSv1_2

# Default to verifying SSL/TLS certs because "safe by default" is a good idea.
DEFAULT_CERT_VERIFY = True

# Default number of times to retry on transient network errors like connection
# timeouts or DNS timeouts.
DEFAULT_RETRIES = 5


[docs]def get_transport_factory(verify=None, ssl_version=None, retries=None): """Get a new callable that returns :class:`requests.Session` instances that have been configured according to the given parameters. :class:`requests.Session` instances are then used for interacting with the API of the load balancer over HTTP or HTTPS. .. versionchanged:: 0.10.0 Using the requests/urllib3 default is no longer an option. Passing a ``None`` value for ``ssl_version`` will result in using the Warthog default (TLS v1.2). .. versionchanged:: 2.0.0 Added the ``retries`` parameter and default it to a number greater than zero. :param bool|None verify: Should SSL certificates by verified when connecting over HTTPS? Default is ``True``. If you have chosen not to verify certificates warnings about this emitted by the requests library will be suppressed. :param int|None ssl_version: Explicit version of SSL to use for HTTPS connections to an A10 load balancer. The version is a constant as specified by the :mod:`ssl` module. The default is TLSv1.2. :param int|None retries: The maximum number of times to retry operations on transient network errors. Note this only applies to cases where we haven't yet sent any data to the server (e.g. connection errors, DNS errors, etc.) :return: A callable to return new configured session instances for making HTTP(S) requests :rtype: callable """ # Using `None` here to represent "not specified" so we don't have to litter the # whole lib with references to the default values we've specified here. Callers # just pass `None` and we'll pick the default here. verify = verify if verify is not None else DEFAULT_CERT_VERIFY ssl_version = ssl_version if ssl_version is not None else DEFAULT_SSL_VERSION retries = retries if retries is not None else DEFAULT_RETRIES # pylint: disable=missing-docstring def factory(): transport = requests.Session() transport.mount('https://', VersionedSSLAdapter(ssl_version, max_retries=retries)) if not verify: transport.verify = False transport.mount('http://', HTTPAdapter( max_retries=retries, pool_connections=DEFAULT_POOLSIZE, pool_maxsize=DEFAULT_POOLSIZE, pool_block=DEFAULT_POOLBLOCK )) return transport # Make sure that we suppress warnings about invalid certs since the user # has explicitly asked us to not verify it, they know that we're doing # something dangerous and don't care. if not verify: warnings.filterwarnings("ignore", category=InsecureRequestWarning) return factory
class VersionedSSLAdapter(HTTPAdapter): """"Transport adapter that requires the use of a specific version of SSL.""" # pylint: disable=too-many-arguments def __init__(self, ssl_version, pool_connections=DEFAULT_POOLSIZE, pool_maxsize=DEFAULT_POOLSIZE, max_retries=DEFAULT_RETRIES, pool_block=DEFAULT_POOLBLOCK): self.ssl_version = ssl_version super(VersionedSSLAdapter, self).__init__( pool_connections=pool_connections, pool_maxsize=pool_maxsize, max_retries=max_retries, pool_block=pool_block ) def init_poolmanager(self, connections, maxsize, block=False, **pool_kwargs): # pylint: disable=attribute-defined-outside-init self.poolmanager = PoolManager( num_pools=connections, maxsize=maxsize, block=block, ssl_version=self.ssl_version, **pool_kwargs)