Source code for ironic_lib.capabilities
#    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.
"""Code for working with capabilities."""
import json
import logging
from ironic_lib.common.i18n import _
LOG = logging.getLogger(__name__)
def _parse_old_format(cap_str, skip_malformed=True):
    """Extract capabilities from string.
    :param cap_str: A string in the key1:value1,key2:value2 format.
    :param skip_malformed: Whether to skip malformed items or raise ValueError.
    :return: a dictionary
    """
    capabilities = {}
    for node_capability in cap_str.split(','):
        parts = node_capability.split(':', 1)
        if len(parts) == 2 and parts[0] and parts[1]:
            capabilities[parts[0]] = parts[1]
        else:
            if skip_malformed:
                LOG.warning("Ignoring malformed capability '%s'. "
                            "Format should be 'key:val'.", node_capability)
            else:
                raise ValueError(
                    _("Malformed capability %s. Format should be 'key:val'")
                    % node_capability)
    return capabilities
[docs]def parse(capabilities, compat=True, skip_malformed=False):
    """Extract capabilities from provided object.
    The capabilities value can either be a dict, or a json str, or
    a key1:value1,key2:value2 formatted string (if compat is True).
    If None, an empty dictionary is returned.
    :param capabilities: The capabilities value. Can either be a dict, or
                         a json str, or a key1:value1,key2:value2 formatted
                         string (if compat is True).
    :param compat: Whether to parse the old format key1:value1,key2:value2.
    :param skip_malformed: Whether to skip malformed items or raise ValueError.
    :returns: A dictionary with the capabilities if found and well formatted,
              otherwise an empty dictionary.
    :raises: TypeError if the capabilities are of invalid type.
    :raises: ValueError on a malformed capability if skip_malformed is False
             or on invalid JSON with compat is False.
    """
    if capabilities is None:
        return {}
    elif isinstance(capabilities, str):
        try:
            return json.loads(capabilities)
        except (ValueError, TypeError) as exc:
            if compat:
                return _parse_old_format(capabilities,
                                         skip_malformed=skip_malformed)
            else:
                raise ValueError(
                    _('Invalid JSON capabilities %(value)s: %(error)s')
                    % {'value': capabilities, 'error': exc})
    elif not isinstance(capabilities, dict):
        raise TypeError(
            _('Invalid capabilities, expected a string or a dict, got %s')
            % capabilities)
    else:
        return capabilities 
[docs]def combine(capabilities_dict, skip_none=False):
    """Combine capabilities into the old format.
    :param capabilities_dict: Capabilities as a mapping.
    :param skip_none: If True, skips all items with value of None.
    :returns: Capabilities as a string key1:value1,key2:value2.
    """
    return ','.join(["%s:%s" % (key, value)
                     for key, value in capabilities_dict.items()
                     if not skip_none or value is not None]) 
[docs]def update_and_combine(capabilities, new_values, skip_malformed=False,
                       skip_none=False):
    """Parses capabilities, updated them with new values and re-combines.
    :param capabilities: The capabilities value. Can either be a dict, or
                         a json str, or a key1:value1,key2:value2 formatted
                         string (if compat is True).
    :param new_values: New values as a dictionary.
    :param skip_malformed: Whether to skip malformed items or raise ValueError.
    :param skip_none: If True, skips all items with value of None.
    :returns: Capabilities in the old format (key1:value1,key2:value2).
    :raises: TypeError if the capabilities are of invalid type.
    :raises: ValueError on a malformed capability if skip_malformed is False.
    """
    if not isinstance(new_values, dict):
        raise TypeError(
            _("Cannot update capabilities. The new capabilities should be in "
              "a dictionary. Provided value is %s") % new_values)
    capabilities = parse(capabilities, skip_malformed=skip_malformed)
    capabilities.update(new_values)
    return combine(capabilities, skip_none=skip_none)