2016-12-08 13:37:36 +00:00
|
|
|
# Copyright 2017 Cisco Systems
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
|
|
# not use this file except in compliance with the License. You may obtain
|
|
|
|
# a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
# License for the specific language governing permissions and limitations
|
|
|
|
# under the License.
|
|
|
|
|
|
|
|
import logging
|
|
|
|
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
# A list of features and their supported microversions. Note that these are
|
|
|
|
# explicit functioning versions, not a range.
|
|
|
|
# There should be a minimum of two versions per feature. The first entry in
|
|
|
|
# this list should always be the lowest possible API microversion for a
|
|
|
|
# feature i.e. the version at which that feature was introduced. The second
|
|
|
|
# entry should be the current service version when the feature was added to
|
|
|
|
# horizon.
|
|
|
|
# Further documentation can be found at
|
2017-09-07 14:45:02 +08:00
|
|
|
# https://docs.openstack.org/horizon/latest/contributor/topics/
|
|
|
|
# microversion_support.html
|
2016-12-08 13:37:36 +00:00
|
|
|
MICROVERSION_FEATURES = {
|
|
|
|
"nova": {
|
2016-04-19 08:10:23 +08:00
|
|
|
"locked_attribute": ["2.9", "2.42"],
|
2021-06-29 23:38:41 +05:30
|
|
|
"instance_description": ["2.19", "2.60"],
|
2017-11-04 14:29:14 +08:00
|
|
|
"remote_console_mks": ["2.8", "2.53"],
|
2017-11-05 22:50:41 +08:00
|
|
|
"servergroup_soft_policies": ["2.15", "2.60"],
|
2018-02-25 14:38:43 -05:00
|
|
|
"servergroup_user_info": ["2.13", "2.60"],
|
2017-11-18 17:36:39 +00:00
|
|
|
"multiattach": ["2.60"],
|
2021-06-29 23:38:41 +05:30
|
|
|
"auto_allocated_network": ["2.37", "2.60"],
|
2019-02-19 15:29:49 +02:00
|
|
|
"key_types": ["2.2", "2.9"],
|
|
|
|
"key_type_list": ["2.9"],
|
2022-09-12 15:04:50 +02:00
|
|
|
"rescue_instance_volume_based": ["2.87", "2.93"],
|
2016-12-08 13:37:36 +00:00
|
|
|
},
|
|
|
|
"cinder": {
|
2019-02-04 08:35:36 +00:00
|
|
|
"groups": ["3.27", "3.43", "3.48", "3.58"],
|
2019-01-15 03:34:23 +09:00
|
|
|
"message_list": ["3.5", "3.29"],
|
|
|
|
"limits_project_id_query": ["3.43", "3.50", "3.55"],
|
2020-08-31 11:42:05 +02:00
|
|
|
"extend_in_use_volume": ["3.42", "3.60"],
|
2016-12-08 13:37:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-01-15 04:23:08 +09:00
|
|
|
class MicroVersionNotFound(Exception):
|
|
|
|
def __init__(self, features):
|
|
|
|
self.features = features
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return "Insufficient microversion for %s" % self.features
|
|
|
|
|
|
|
|
|
|
|
|
def get_requested_versions(service, features):
|
2017-10-22 18:17:03 +00:00
|
|
|
if not features:
|
|
|
|
return None
|
2018-06-11 14:11:25 +08:00
|
|
|
# Convert a single feature string into a list for backward compatibility.
|
2017-10-22 18:17:03 +00:00
|
|
|
if isinstance(features, str):
|
|
|
|
features = [features]
|
2016-12-08 13:37:36 +00:00
|
|
|
try:
|
|
|
|
service_features = MICROVERSION_FEATURES[service]
|
|
|
|
except KeyError:
|
2017-04-11 10:24:01 +00:00
|
|
|
LOG.debug("'%s' could not be found in the MICROVERSION_FEATURES dict",
|
|
|
|
service)
|
2016-12-08 13:37:36 +00:00
|
|
|
return None
|
2017-10-22 18:17:03 +00:00
|
|
|
|
|
|
|
feature_versions = set(service_features[features[0]])
|
|
|
|
for feature in features[1:]:
|
|
|
|
feature_versions &= set(service_features[feature])
|
|
|
|
if not feature_versions:
|
|
|
|
return None
|
|
|
|
# Sort version candidates from larger versins
|
|
|
|
feature_versions = sorted(feature_versions, reverse=True,
|
|
|
|
key=lambda v: [int(i) for i in v.split('.')])
|
2019-01-15 04:23:08 +09:00
|
|
|
return feature_versions
|
|
|
|
|
|
|
|
|
|
|
|
# NOTE(robcresswell): Since each client implements their own wrapper class for
|
|
|
|
# API objects, we'll need to allow that to be passed in. In the future this
|
|
|
|
# should be replaced by some common handling in Oslo.
|
|
|
|
def get_microversion_for_features(service, features, wrapper_class,
|
|
|
|
min_ver, max_ver):
|
|
|
|
"""Retrieves that highest known functional microversion for features"""
|
|
|
|
feature_versions = get_requested_versions(service, features)
|
|
|
|
if not feature_versions:
|
|
|
|
return None
|
|
|
|
|
2017-10-22 18:17:03 +00:00
|
|
|
for version in feature_versions:
|
2016-12-08 13:37:36 +00:00
|
|
|
microversion = wrapper_class(version)
|
|
|
|
if microversion.matches(min_ver, max_ver):
|
|
|
|
return microversion
|
|
|
|
return None
|