Dell PowerStore: Added timeout into rest API call.

Added connect and read timeout into rest API call
of PowerStore to avoid cinder hang issue.
Deafult value of connect and read timeout
is 30 seconds.

Closes-Bug: #2055022
Change-Id: Id44a4b01c9ca238470e7d984d39b5e4c4883fd6a
This commit is contained in:
Nilesh Thathagar 2024-03-07 11:06:39 +05:30
parent a72c4894e9
commit 897980b252
7 changed files with 144 additions and 32 deletions

View File

@ -17,6 +17,7 @@ from unittest import mock
import uuid import uuid
import ddt import ddt
import requests.exceptions
from cinder import exception from cinder import exception
from cinder.tests.unit import test from cinder.tests.unit import test
@ -31,7 +32,9 @@ CLIENT_OPTIONS = {
"rest_username": "fake_user", "rest_username": "fake_user",
"rest_password": "fake_password", "rest_password": "fake_password",
"verify_certificate": False, "verify_certificate": False,
"certificate_path": None "certificate_path": None,
"rest_api_connect_timeout": 60,
"rest_api_read_timeout": 60
} }
ISCSI_IP_POOL_RESP = [ ISCSI_IP_POOL_RESP = [
@ -277,3 +280,52 @@ class TestClient(test.TestCase):
self.client.update_qos_io_rule, self.client.update_qos_io_rule,
"io-rule-6b6e5489-4b5b-4468-a1f7-32cec2ffa3bf", "io-rule-6b6e5489-4b5b-4468-a1f7-32cec2ffa3bf",
QOS_UPDATE_IO_RULE_PARAMS) QOS_UPDATE_IO_RULE_PARAMS)
@mock.patch("requests.request")
def test_get_request_timeout_exception(self, mock_request):
mock_request.return_value = MockResponse(
rc=501
)
error = self.assertRaises(
exception.VolumeBackendAPIException,
self.client.get_array_version)
self.assertEqual('Bad or unexpected response from the '
'storage volume backend API: Failed to '
'query PowerStore array version.',
error.msg)
@mock.patch("requests.request")
def test_send_get_request_connect_timeout_exception(self,
mock_request):
mock_request.side_effect = requests.exceptions.ConnectTimeout()
r, resp = self.client._send_request("GET",
"/api/version")
self.assertEqual(500, r.status_code)
@mock.patch("requests.request")
def test_send_get_request_read_timeout_exception(self,
mock_request):
mock_request.side_effect = requests.exceptions.ReadTimeout()
r, resp = self.client._send_request("GET",
"/api/version")
self.assertEqual(500, r.status_code)
@mock.patch("requests.request")
def test_send_post_request_connect_timeout_exception(self,
mock_request):
params = {}
mock_request.side_effect = requests.exceptions.ConnectTimeout()
r, resp = self.client._send_request("POST",
"/metrics/generate",
params)
self.assertEqual(500, r.status_code)
@mock.patch("requests.request")
def test_send_post_request_read_timeout_exception(self,
mock_request):
params = {}
mock_request.side_effect = requests.exceptions.ReadTimeout()
r, resp = self.client._send_request("POST",
"/metrics/generate",
params)
self.assertEqual(500, r.status_code)

View File

@ -157,3 +157,17 @@ class TestVolumeCreateDeleteExtend(powerstore.TestPowerStoreDriver):
self.volume, self.volume,
16) 16)
self.assertIn("Failed to extend PowerStore volume", error.msg) self.assertIn("Failed to extend PowerStore volume", error.msg)
@mock.patch("requests.request")
def test_post_request_timeout_exception(self, mock_request):
mock_request.return_value = powerstore.MockResponse(
rc=501
)
error = self.assertRaises(exception.VolumeBackendAPIException,
self.driver.extend_volume,
self.volume,
16)
self.assertEqual('Bad or unexpected response from the '
'storage volume backend API: Failed to '
'extend PowerStore volume with id fake_id.',
error.msg)

View File

@ -21,6 +21,7 @@ import json
from oslo_log import log as logging from oslo_log import log as logging
from oslo_utils import strutils from oslo_utils import strutils
import requests import requests
import requests.exceptions
from cinder import exception from cinder import exception
from cinder.i18n import _ from cinder.i18n import _
@ -45,7 +46,9 @@ class PowerStoreClient(object):
rest_username, rest_username,
rest_password, rest_password,
verify_certificate, verify_certificate,
certificate_path): certificate_path,
rest_api_connect_timeout,
rest_api_read_timeout):
self.rest_ip = rest_ip self.rest_ip = rest_ip
self.rest_username = rest_username self.rest_username = rest_username
self.rest_password = rest_password self.rest_password = rest_password
@ -59,6 +62,8 @@ class PowerStoreClient(object):
requests.codes.no_content, requests.codes.no_content,
requests.codes.partial_content requests.codes.partial_content
] ]
self.rest_api_connect_timeout = rest_api_connect_timeout
self.rest_api_read_timeout = rest_api_read_timeout
@property @property
def _verify_cert(self): def _verify_cert(self):
@ -92,6 +97,9 @@ class PowerStoreClient(object):
payload=None, payload=None,
params=None, params=None,
log_response_data=True): log_response_data=True):
response = None
r = requests.Response
try:
if not params: if not params:
params = {} params = {}
request_params = { request_params = {
@ -102,8 +110,10 @@ class PowerStoreClient(object):
if payload and method != "GET": if payload and method != "GET":
request_params["data"] = json.dumps(payload) request_params["data"] = json.dumps(payload)
request_url = self.base_url + url request_url = self.base_url + url
r = requests.request(method, request_url, **request_params) timeout = (self.rest_api_connect_timeout,
self.rest_api_read_timeout)
r = requests.request(method, request_url, **request_params,
timeout=timeout)
log_level = logging.DEBUG log_level = logging.DEBUG
if r.status_code not in self.ok_codes: if r.status_code not in self.ok_codes:
log_level = logging.ERROR log_level = logging.ERROR
@ -112,16 +122,21 @@ class PowerStoreClient(object):
r.request.method, r.request.method,
r.request.url, r.request.url,
strutils.mask_password(r.request.body)) strutils.mask_password(r.request.body))
if log_response_data or log_level == logging.ERROR: if (log_response_data or
msg = "REST Response: %s with data %s" % (r.status_code, r.text) log_level == logging.ERROR):
msg = ("REST Response: %s with data %s" %
(r.status_code, r.text))
else: else:
msg = "REST Response: %s" % r.status_code msg = "REST Response: %s" % r.status_code
LOG.log(log_level, msg) LOG.log(log_level, msg)
try: try:
response = r.json() response = r.json()
except ValueError: except ValueError:
response = None response = None
except requests.exceptions.Timeout as e:
r.status_code = requests.codes.internal_server_error
LOG.error("The request to URL %(url)s failed with timeout "
"exception %(exc)s", {"url": url, "exc": e})
return r, response return r, response
_send_get_request = functools.partialmethod(_send_request, "GET") _send_get_request = functools.partialmethod(_send_request, "GET")

View File

@ -266,4 +266,8 @@ class PowerStoreDriver(driver.VolumeDriver):
conf["rest_password"] = get_value("san_password") conf["rest_password"] = get_value("san_password")
conf["verify_certificate"] = get_value("driver_ssl_cert_verify") conf["verify_certificate"] = get_value("driver_ssl_cert_verify")
conf["certificate_path"] = get_value("driver_ssl_cert_path") conf["certificate_path"] = get_value("driver_ssl_cert_path")
conf["rest_api_connect_timeout"] = (
self.configuration.safe_get(utils.POWERSTORE_REST_CONNECT_TIMEOUT))
conf["rest_api_read_timeout"] = (
self.configuration.safe_get(utils.POWERSTORE_REST_READ_TIMEOUT))
return conf return conf

View File

@ -17,6 +17,8 @@
from oslo_config import cfg from oslo_config import cfg
from cinder.volume.drivers.dell_emc.powerstore import utils as store_utils
POWERSTORE_APPLIANCES = "powerstore_appliances" POWERSTORE_APPLIANCES = "powerstore_appliances"
POWERSTORE_PORTS = "powerstore_ports" POWERSTORE_PORTS = "powerstore_ports"
POWERSTORE_NVME = "powerstore_nvme" POWERSTORE_NVME = "powerstore_nvme"
@ -39,5 +41,16 @@ POWERSTORE_OPTS = [
), ),
cfg.BoolOpt(POWERSTORE_NVME, cfg.BoolOpt(POWERSTORE_NVME,
default=False, default=False,
help="Connect PowerStore volumes using NVMe-OF.") help="Connect PowerStore volumes using NVMe-OF."),
cfg.IntOpt(store_utils.POWERSTORE_REST_CONNECT_TIMEOUT,
default=30, min=1,
help='Use this value to specify the connect '
'timeout value (in seconds) for REST API calls '
'to the PowerStore backend.'),
cfg.IntOpt(store_utils.POWERSTORE_REST_READ_TIMEOUT,
default=30, min=1,
help='Use this value to specify the read '
'timeout value (in seconds) for REST API calls '
'to the PowerStore backend.')
] ]

View File

@ -38,6 +38,8 @@ PROTOCOL_NVME = "NVMe"
POWERSTORE_PP_KEY = "powerstore:protection_policy" POWERSTORE_PP_KEY = "powerstore:protection_policy"
VOLUME_ATTACH_OPERATION = 1 VOLUME_ATTACH_OPERATION = 1
VOLUME_DETACH_OPERATION = 2 VOLUME_DETACH_OPERATION = 2
POWERSTORE_REST_CONNECT_TIMEOUT = "rest_api_call_connect_timeout"
POWERSTORE_REST_READ_TIMEOUT = "rest_api_call_read_timeout"
def bytes_to_gib(size_in_bytes): def bytes_to_gib(size_in_bytes):

View File

@ -0,0 +1,12 @@
---
fixes:
- |
Dell PowerStore Driver `bug #2055022
<https://bugs.launchpad.net/cinder/+bug/2055022>`_: REST
API calls to the PowerStore backend did not have a timeout
set, which could result in cinder waiting forever.
This fix introduces two configuration options,
``rest_api_call_connect_timeout`` and
``rest_api_call_read_timeout``, to control timeouts
when connecting to the backend.
The default value of each is 30 seconds.