From 5ab3438f93322ed6bae4d33abb6a0a805e04d5c0 Mon Sep 17 00:00:00 2001 From: Mehdi Abaakouk Date: Fri, 27 Apr 2018 15:42:54 +0200 Subject: [PATCH] swift: Allow to configure the requests pool size By default, python-requests share a pool for the whole process and this one have only 10 parallels connections. This change use the keystoneauth1 adapter. So, we have a pool dedicated for swift and we can configure it. And we also get all tcp keepalive workaround for free. This allows keep the default for 'retries'. Closes: #509 --- gnocchi/common/swift.py | 69 ++++++++++++++++++++++++---------------- gnocchi/storage/swift.py | 5 +++ 2 files changed, 47 insertions(+), 27 deletions(-) diff --git a/gnocchi/common/swift.py b/gnocchi/common/swift.py index f961ba441..0c2ab82b9 100644 --- a/gnocchi/common/swift.py +++ b/gnocchi/common/swift.py @@ -12,46 +12,61 @@ # License for the specific language governing permissions and limitations # under the License. import daiquiri +import keystoneauth1.session from six.moves.urllib.parse import quote +from gnocchi import storage + +LOG = daiquiri.getLogger(__name__) + + try: from swiftclient import client as swclient from swiftclient import utils as swift_utils + + class CustomSwiftConnecion(swclient.Connection): + def __init__(self, conf): + self.conf = conf + + os_options = { + 'endpoint_type': conf.swift_endpoint_type, + 'service_type': conf.swift_service_type, + 'user_domain_name': conf.swift_user_domain_name, + 'project_domain_name': conf.swift_project_domain_name, + } + if conf.swift_region: + os_options['region_name'] = conf.swift_region + + super(CustomSwiftConnecion, self).__init__( + preauthurl=conf.swift_url, + auth_version=conf.swift_auth_version, + authurl=conf.swift_authurl, + preauthtoken=conf.swift_preauthtoken, + user=conf.swift_user, + key=conf.swift_key, + tenant_name=conf.swift_project_name, + timeout=conf.swift_timeout, + insecure=conf.swift_auth_insecure, + os_options=os_options, + cacert=conf.swift_cacert) + + def http_connection(self, url=None): + url, conn = super(CustomSwiftConnecion, self).http_connection(url) + adapter = keystoneauth1.session.TCPKeepAliveAdapter( + pool_maxsize=self.conf.swift_max_parallel_requests) + conn.request_session.mount("http://", adapter) + conn.request_session.mount("https://", adapter) + return url, conn + except ImportError: swclient = None swift_utils = None -from gnocchi import storage - -LOG = daiquiri.getLogger(__name__) - def get_connection(conf): if swclient is None: raise RuntimeError("python-swiftclient unavailable") - - os_options = { - 'endpoint_type': conf.swift_endpoint_type, - 'service_type': conf.swift_service_type, - 'user_domain_name': conf.swift_user_domain_name, - 'project_domain_name': conf.swift_project_domain_name, - } - if conf.swift_region: - os_options['region_name'] = conf.swift_region - - return swclient.Connection( - preauthurl=conf.swift_url, - auth_version=conf.swift_auth_version, - authurl=conf.swift_authurl, - preauthtoken=conf.swift_preauthtoken, - user=conf.swift_user, - key=conf.swift_key, - tenant_name=conf.swift_project_name, - timeout=conf.swift_timeout, - insecure=conf.swift_auth_insecure, - os_options=os_options, - cacert=conf.swift_cacert, - retries=0) + return CustomSwiftConnecion(conf) POST_HEADERS = {'Accept': 'application/json', 'Content-Type': 'text/plain'} diff --git a/gnocchi/storage/swift.py b/gnocchi/storage/swift.py index c1a1ff65a..b21b0b4ec 100644 --- a/gnocchi/storage/swift.py +++ b/gnocchi/storage/swift.py @@ -38,6 +38,11 @@ cfg.StrOpt('swift_url', help='Swift URL. ' 'If unset, it is obtained from the auth service.'), + cfg.IntOpt('swift_max_parallel_requests', + default=50, + # NOTE(sileht): python-requests default in 10, it's far too low + # for any workload. + help='Maximun of Swift parallel requests.'), cfg.StrOpt('swift_authurl', default="http://localhost:8080/auth/v1.0", help='Swift auth URL.'),