Merge "NetApp SolidFire: Fix misbehavior on account creation"
This commit is contained in:
commit
6f2cba522c
@ -118,9 +118,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
self.ctxt, volume_id=self.vol.id)
|
self.ctxt, volume_id=self.vol.id)
|
||||||
|
|
||||||
self.fake_sfaccount = {'accountID': 25,
|
self.fake_sfaccount = {'accountID': 25,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne',
|
'username': 'prefix-testprjid',
|
||||||
'volumes': [6, 7, 20]}
|
'volumes': [6, 7, 20]}
|
||||||
|
|
||||||
self.fake_sfvol = {'volumeID': 6,
|
self.fake_sfvol = {'volumeID': 6,
|
||||||
@ -304,6 +303,16 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
"snapshotID": 1
|
"snapshotID": 1
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
elif method is 'ListAccounts':
|
||||||
|
return {
|
||||||
|
'result': {
|
||||||
|
'accounts': [{
|
||||||
|
'accountID': 5,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid'
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
}
|
||||||
else:
|
else:
|
||||||
# Crap, unimplemented API call in Fake
|
# Crap, unimplemented API call in Fake
|
||||||
return None
|
return None
|
||||||
@ -351,9 +360,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
'created_at': timeutils.utcnow()}
|
'created_at': timeutils.utcnow()}
|
||||||
|
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
|
|
||||||
test_type = {'name': 'sf-1',
|
test_type = {'name': 'sf-1',
|
||||||
'qos_specs_id': 'fb0576d7-b4b5-4cad-85dc-ca92e6a497d1',
|
'qos_specs_id': 'fb0576d7-b4b5-4cad-85dc-ca92e6a497d1',
|
||||||
@ -415,9 +423,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
'volume_type_id': None,
|
'volume_type_id': None,
|
||||||
'created_at': timeutils.utcnow()}
|
'created_at': timeutils.utcnow()}
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
|
|
||||||
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
with mock.patch.object(sfv,
|
with mock.patch.object(sfv,
|
||||||
@ -445,9 +452,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
'volume_type_id': None,
|
'volume_type_id': None,
|
||||||
'created_at': timeutils.utcnow()}
|
'created_at': timeutils.utcnow()}
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
|
|
||||||
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
with mock.patch.object(sfv,
|
with mock.patch.object(sfv,
|
||||||
@ -487,7 +493,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
mock.patch.object(sfv,
|
mock.patch.object(sfv,
|
||||||
'_get_sfaccounts_for_tenant',
|
'_get_sfaccounts_for_tenant',
|
||||||
return_value=[{'accountID': 5,
|
return_value=[{'accountID': 5,
|
||||||
'name': 'testprjid'}]):
|
'username':
|
||||||
|
'prefix-testprjid'}]):
|
||||||
sfv.create_snapshot(testsnap)
|
sfv.create_snapshot(testsnap)
|
||||||
sfv.delete_snapshot(testsnap)
|
sfv.delete_snapshot(testsnap)
|
||||||
|
|
||||||
@ -587,7 +594,7 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
self.mock_object(solidfire.SolidFireDriver,
|
self.mock_object(solidfire.SolidFireDriver,
|
||||||
'_issue_api_request',
|
'_issue_api_request',
|
||||||
self.fake_issue_api_request)
|
self.fake_issue_api_request)
|
||||||
account = sfv._create_sfaccount('project-id')
|
account = sfv._create_sfaccount('some-name')
|
||||||
self.assertIsNotNone(account)
|
self.assertIsNotNone(account)
|
||||||
|
|
||||||
def test_create_sfaccount_fails(self):
|
def test_create_sfaccount_fails(self):
|
||||||
@ -598,6 +605,22 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
self.assertRaises(exception.SolidFireAPIException,
|
self.assertRaises(exception.SolidFireAPIException,
|
||||||
sfv._create_sfaccount, 'project-id')
|
sfv._create_sfaccount, 'project-id')
|
||||||
|
|
||||||
|
def test_get_sfaccounts_for_tenant(self):
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
self.mock_object(solidfire.SolidFireDriver,
|
||||||
|
'_issue_api_request',
|
||||||
|
self.fake_issue_api_request)
|
||||||
|
accounts = sfv._get_sfaccounts_for_tenant('some-name')
|
||||||
|
self.assertIsNotNone(accounts)
|
||||||
|
|
||||||
|
def test_get_sfaccounts_for_tenant_fails(self):
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
self.mock_object(solidfire.SolidFireDriver,
|
||||||
|
'_issue_api_request',
|
||||||
|
self.fake_issue_api_request_fails)
|
||||||
|
self.assertRaises(exception.SolidFireAPIException,
|
||||||
|
sfv._get_sfaccounts_for_tenant, 'some-name')
|
||||||
|
|
||||||
def test_get_sfaccount_by_name(self):
|
def test_get_sfaccount_by_name(self):
|
||||||
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
self.mock_object(solidfire.SolidFireDriver,
|
self.mock_object(solidfire.SolidFireDriver,
|
||||||
@ -606,6 +629,80 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
account = sfv._get_sfaccount_by_name('some-name')
|
account = sfv._get_sfaccount_by_name('some-name')
|
||||||
self.assertIsNotNone(account)
|
self.assertIsNotNone(account)
|
||||||
|
|
||||||
|
def test_get_account_create_availability_no_account(self):
|
||||||
|
fake_sfaccounts = []
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
sfaccount = sfv._get_account_create_availability(fake_sfaccounts)
|
||||||
|
self.assertIsNone(sfaccount)
|
||||||
|
|
||||||
|
def test_get_account_create_availability(self):
|
||||||
|
fake_sfaccounts = [{'accountID': 29,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid',
|
||||||
|
'volumes': [6, 7, 20]}]
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
sfaccount = sfv._get_account_create_availability(fake_sfaccounts)
|
||||||
|
self.assertIsNotNone(sfaccount)
|
||||||
|
self.assertEqual(sfaccount['accountID'],
|
||||||
|
fake_sfaccounts[0]['accountID'])
|
||||||
|
|
||||||
|
def test_get_account_create_availability_primary_full(self):
|
||||||
|
fake_sfaccounts = [{'accountID': 30,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid'}]
|
||||||
|
get_sfaccount_result = {'accountID': 31,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid_'}
|
||||||
|
get_vol_result = list(range(1, 2001))
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
with mock.patch.object(sfv,
|
||||||
|
'_get_sfaccounts_for_tenant',
|
||||||
|
return_value=fake_sfaccounts), \
|
||||||
|
mock.patch.object(sfv,
|
||||||
|
'_get_volumes_for_account',
|
||||||
|
return_value=get_vol_result):
|
||||||
|
sfaccount = sfv._get_account_create_availability(fake_sfaccounts)
|
||||||
|
self.assertIsNotNone(sfaccount)
|
||||||
|
self.assertEqual(sfaccount['username'],
|
||||||
|
get_sfaccount_result['username'])
|
||||||
|
|
||||||
|
def test_get_account_create_availability_both_full(self):
|
||||||
|
fake_sfaccounts = [{'accountID': 32,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid'},
|
||||||
|
{'accountID': 33,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid_'}]
|
||||||
|
get_vol_result = list(range(1, 2001))
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
with mock.patch.object(sfv,
|
||||||
|
'_get_sfaccounts_for_tenant',
|
||||||
|
return_value=fake_sfaccounts), \
|
||||||
|
mock.patch.object(sfv,
|
||||||
|
'_get_volumes_for_account',
|
||||||
|
return_value=get_vol_result):
|
||||||
|
sfaccount = sfv._get_account_create_availability(fake_sfaccounts)
|
||||||
|
self.assertIsNone(sfaccount)
|
||||||
|
|
||||||
|
def test_get_create_account(self):
|
||||||
|
fake_sfaccounts = [{'accountID': 34,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid'},
|
||||||
|
{'accountID': 35,
|
||||||
|
'targetSecret': 'shhhh',
|
||||||
|
'username': 'prefix-testprjid_'}]
|
||||||
|
get_vol_result = list(range(1, 2001))
|
||||||
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
|
with mock.patch.object(sfv,
|
||||||
|
'_get_sfaccounts_for_tenant',
|
||||||
|
return_value=fake_sfaccounts), \
|
||||||
|
mock.patch.object(sfv,
|
||||||
|
'_get_volumes_for_account',
|
||||||
|
return_value=get_vol_result):
|
||||||
|
sfaccount = sfv._get_account_create_availability(fake_sfaccounts)
|
||||||
|
self.assertRaises(exception.SolidFireDriverException,
|
||||||
|
sfv._get_create_account, sfaccount)
|
||||||
|
|
||||||
def test_get_sfaccount_by_name_fails(self):
|
def test_get_sfaccount_by_name_fails(self):
|
||||||
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
sfv = solidfire.SolidFireDriver(configuration=self.configuration)
|
||||||
self.mock_object(solidfire.SolidFireDriver,
|
self.mock_object(solidfire.SolidFireDriver,
|
||||||
@ -616,9 +713,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_get_sfvol_by_cinder_vref_no_provider_id(self):
|
def test_get_sfvol_by_cinder_vref_no_provider_id(self):
|
||||||
fake_sfaccounts = [{'accountID': 25,
|
fake_sfaccounts = [{'accountID': 25,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne',
|
'username': 'prefix-testprjid',
|
||||||
'volumes': [6, 7, 20]}]
|
'volumes': [6, 7, 20]}]
|
||||||
self.mock_vref = mock_vref()
|
self.mock_vref = mock_vref()
|
||||||
|
|
||||||
@ -651,9 +747,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_get_sfvol_by_cinder_vref_no_provider_id_nomatch(self):
|
def test_get_sfvol_by_cinder_vref_no_provider_id_nomatch(self):
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne',
|
'username': 'prefix-testprjid',
|
||||||
'volumes': [5, 6, 7, 8]}]
|
'volumes': [5, 6, 7, 8]}]
|
||||||
|
|
||||||
self.mock_vref = mock_vref()
|
self.mock_vref = mock_vref()
|
||||||
@ -673,9 +768,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_get_sfvol_by_cinder_vref_nomatch(self):
|
def test_get_sfvol_by_cinder_vref_nomatch(self):
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne',
|
'username': 'prefix-testprjid',
|
||||||
'volumes': [5, 6, 7, 8]}]
|
'volumes': [5, 6, 7, 8]}]
|
||||||
|
|
||||||
self.mock_vref = mock_vref()
|
self.mock_vref = mock_vref()
|
||||||
@ -695,9 +789,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_get_sfvol_by_cinder_vref(self):
|
def test_get_sfvol_by_cinder_vref(self):
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne',
|
'username': 'prefix-testprjid',
|
||||||
'volumes': [5, 6, 7, 8]}]
|
'volumes': [5, 6, 7, 8]}]
|
||||||
|
|
||||||
self.mock_vref = mock_vref()
|
self.mock_vref = mock_vref()
|
||||||
@ -777,9 +870,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
multiattach=True)
|
multiattach=True)
|
||||||
|
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
|
|
||||||
get_vol_result = {'volumeID': 5,
|
get_vol_result = {'volumeID': 5,
|
||||||
'name': 'test_volume',
|
'name': 'test_volume',
|
||||||
@ -812,9 +904,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_delete_volume_no_volume_on_backend(self):
|
def test_delete_volume_no_volume_on_backend(self):
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
fake_no_volumes = []
|
fake_no_volumes = []
|
||||||
testvol = test_utils.create_volume(self.ctxt)
|
testvol = test_utils.create_volume(self.ctxt)
|
||||||
|
|
||||||
@ -829,9 +920,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
|
|
||||||
def test_delete_snapshot_no_snapshot_on_backend(self):
|
def test_delete_snapshot_no_snapshot_on_backend(self):
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
fake_no_volumes = []
|
fake_no_volumes = []
|
||||||
testvol = test_utils.create_volume(
|
testvol = test_utils.create_volume(
|
||||||
self.ctxt,
|
self.ctxt,
|
||||||
@ -1142,9 +1232,8 @@ class SolidFireVolumeTestCase(test.TestCase):
|
|||||||
'migration_status': 'target:'
|
'migration_status': 'target:'
|
||||||
'a720b3c0-d1f0-11e1-9b23-0800200c9a66'}
|
'a720b3c0-d1f0-11e1-9b23-0800200c9a66'}
|
||||||
fake_sfaccounts = [{'accountID': 5,
|
fake_sfaccounts = [{'accountID': 5,
|
||||||
'name': 'testprjid',
|
|
||||||
'targetSecret': 'shhhh',
|
'targetSecret': 'shhhh',
|
||||||
'username': 'john-wayne'}]
|
'username': 'prefix-testprjid'}]
|
||||||
|
|
||||||
def _fake_do_v_create(project_id, params):
|
def _fake_do_v_create(project_id, params):
|
||||||
return project_id, params
|
return project_id, params
|
||||||
|
@ -647,7 +647,7 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
|
|
||||||
return sfaccount
|
return sfaccount
|
||||||
|
|
||||||
def _create_sfaccount(self, project_id):
|
def _create_sfaccount(self, sf_account_name):
|
||||||
"""Create account on SolidFire device if it doesn't already exist.
|
"""Create account on SolidFire device if it doesn't already exist.
|
||||||
|
|
||||||
We're first going to check if the account already exists, if it does
|
We're first going to check if the account already exists, if it does
|
||||||
@ -655,7 +655,6 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
sf_account_name = self._get_sf_account_name(project_id)
|
|
||||||
sfaccount = self._get_sfaccount_by_name(sf_account_name)
|
sfaccount = self._get_sfaccount_by_name(sf_account_name)
|
||||||
if sfaccount is None:
|
if sfaccount is None:
|
||||||
LOG.debug('solidfire account: %s does not exist, create it...',
|
LOG.debug('solidfire account: %s does not exist, create it...',
|
||||||
@ -1102,7 +1101,8 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
# Also: we expect this to be sorted, so we get the primary first
|
# Also: we expect this to be sorted, so we get the primary first
|
||||||
# in the list
|
# in the list
|
||||||
return sorted([acc for acc in accounts if
|
return sorted([acc for acc in accounts if
|
||||||
cinder_project_id in acc['username']])
|
cinder_project_id in acc['username']],
|
||||||
|
key=lambda k: k['accountID'])
|
||||||
|
|
||||||
def _get_all_active_volumes(self, cinder_uuid=None):
|
def _get_all_active_volumes(self, cinder_uuid=None):
|
||||||
params = {}
|
params = {}
|
||||||
@ -1132,19 +1132,21 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
# if it exists and return whichever one has count
|
# if it exists and return whichever one has count
|
||||||
# available.
|
# available.
|
||||||
for acc in accounts:
|
for acc in accounts:
|
||||||
if self._get_volumes_for_account(
|
if len(self._get_volumes_for_account(
|
||||||
acc['accountID']) > self.max_volumes_per_account:
|
acc['accountID'])) < self.max_volumes_per_account:
|
||||||
return acc
|
return acc
|
||||||
if len(accounts) == 1:
|
if len(accounts) == 1:
|
||||||
sfaccount = self._create_sfaccount(accounts[0]['name'] + '_')
|
sfaccount = self._create_sfaccount(accounts[0]['username'] + '_')
|
||||||
return sfaccount
|
return sfaccount
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _get_create_account(self, proj_id):
|
def _get_create_account(self, proj_id):
|
||||||
# Retrieve SolidFire accountID to be used for creating volumes.
|
# Retrieve SolidFire accountID to be used for creating volumes.
|
||||||
sf_accounts = self._get_sfaccounts_for_tenant(proj_id)
|
sf_accounts = self._get_sfaccounts_for_tenant(proj_id)
|
||||||
|
|
||||||
if not sf_accounts:
|
if not sf_accounts:
|
||||||
sf_account = self._create_sfaccount(proj_id)
|
sf_account_name = self._get_sf_account_name(proj_id)
|
||||||
|
sf_account = self._create_sfaccount(sf_account_name)
|
||||||
else:
|
else:
|
||||||
# Check availability for creates
|
# Check availability for creates
|
||||||
sf_account = self._get_account_create_availability(sf_accounts)
|
sf_account = self._get_account_create_availability(sf_accounts)
|
||||||
@ -2050,7 +2052,7 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
if new_project != volume['project_id']:
|
if new_project != volume['project_id']:
|
||||||
# do a create_sfaccount here as this tenant
|
# do a create_sfaccount here as this tenant
|
||||||
# may not exist on the cluster yet
|
# may not exist on the cluster yet
|
||||||
sfaccount = self._create_sfaccount(new_project)
|
sfaccount = self._get_create_account(new_project)
|
||||||
|
|
||||||
params = {
|
params = {
|
||||||
'volumeID': sf_vol['volumeID'],
|
'volumeID': sf_vol['volumeID'],
|
||||||
@ -2118,7 +2120,7 @@ class SolidFireDriver(san.SanISCSIDriver):
|
|||||||
'ListActiveVolumes', params)['result']['volumes']
|
'ListActiveVolumes', params)['result']['volumes']
|
||||||
|
|
||||||
sf_ref = vols[0]
|
sf_ref = vols[0]
|
||||||
sfaccount = self._create_sfaccount(volume['project_id'])
|
sfaccount = self._get_create_account(volume['project_id'])
|
||||||
|
|
||||||
attributes = {}
|
attributes = {}
|
||||||
qos = self._retrieve_qos_setting(volume)
|
qos = self._retrieve_qos_setting(volume)
|
||||||
|
@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
Fix python3 imcompability issues and make SolidFire driver fully compatible
|
||||||
|
with python3.
|
Loading…
x
Reference in New Issue
Block a user