Source code for pywbem.cim_types

#
# (C) Copyright 2003,2004 Hewlett-Packard Development Company, L.P.
# (C) Copyright 2006,2007 Novell, Inc.
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
# Author: Tim Potter <tpot@hp.com>
# Author: Bart Whiteley <bwhiteley@suse.de>
# Author: Ross Peoples <ross.peoples@gmail.com>
#

# pylint: disable=line-too-long
"""
Python classes for representing values of CIM data types, and related
conversion functions.

The following table shows how CIM data types are represented in Python.
Note that some basic CIM data types are represented with built-in Python
types.

========================================  =====================================
CIM data type                             Python type
========================================  =====================================
boolean                                   :class:`py:bool`
char16                                    :term:`string`
string                                    :term:`string`
string (EmbeddedInstance)                 :class:`~pywbem.CIMInstance`
string (EmbeddedObject)                   :class:`~pywbem.CIMInstance`
                                          or :class:`~pywbem.CIMClass`
datetime                                  :class:`~pywbem.CIMDateTime`
reference                                 :class:`~pywbem.CIMInstanceName`
uint8                                     :class:`~pywbem.Uint8`
uint16                                    :class:`~pywbem.Uint16`
uint32                                    :class:`~pywbem.Uint32`
uint64                                    :class:`~pywbem.Uint64`
sint8                                     :class:`~pywbem.Sint8`
sint16                                    :class:`~pywbem.Sint16`
sint32                                    :class:`~pywbem.Sint32`
sint64                                    :class:`~pywbem.Sint64`
real32                                    :class:`~pywbem.Real32`
real64                                    :class:`~pywbem.Real64`
[] (array)                                :class:`py:list`
========================================  =====================================

The CIM NULL value is represented with Python `None` which can be used for any
CIM typed value to represent NULL.

Note that init methods of pywbem classes that take CIM typed values as input
may support Python types in addition to those shown above. For example, the
:class:`~pywbem.CIMProperty` class represents property values of CIM datetime
type internally as :class:`~pywbem.CIMDateTime` objects, but its init method
accepts :class:`py:datetime.timedelta` objects, :class:`py:datetime.datetime`
objects, :term:`string`, in addition to
:class:`~pywbem.CIMDateTime` objects.
"""
# pylint: enable=line-too-long
# Note: When used before module docstrings, Pylint scopes the disable statement
#       to the whole rest of the file, so we need an enable statement.

# This module is meant to be safe for 'import *'.

from __future__ import absolute_import

from datetime import tzinfo, datetime, timedelta
import re
import warnings
import copy
import traceback
import six

from .config import DEBUG_WARNING_ORIGIN, ENFORCE_INTEGER_RANGE
from ._utils import _ensure_unicode, _hash_item, _format, _to_unicode

if six.PY2:
    # pylint: disable=invalid-name,undefined-variable
    _Longint = long  # noqa: F821
else:
    # pylint: disable=invalid-name
    _Longint = int


__all__ = ['cimtype', 'type_from_name', 'MinutesFromUTC', 'CIMType',
           'CIMDateTime', 'CIMInt', 'Uint8', 'Sint8', 'Uint16', 'Sint16',
           'Uint32', 'Sint32', 'Uint64', 'Sint64', 'CIMFloat', 'Real32',
           'Real64']


class _CIMComparisonMixin(object):  # pylint: disable=too-few-public-methods
    """Mixin class providing default implementations for rich comparison
    operators.

    In Python 2, the rich comparison operators (e.g. `__eq__()`) have
    precedence over the traditional comparator method (`_cmp__()`).
    In Python 3, the comparator method (`_cmp__()`) no longer exists.
    Therefore, implementing the rich comparison operators works in both.

    The default implementations delegate to a comparator method `_cmp()`
    implemented by subclasses. This requires that the subclasses can
    define total ordering. (If they cannot, this mixin class cannot be
    used).
    """

    def __eq__(self, other):
        """
        Invoked when two CIM objects are compared with the `==` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        return self._cmp(other) == 0

    def __ne__(self, other):
        """
        Invoked when two CIM objects are compared with the `!=` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        return self._cmp(other) != 0

    def __ordering_deprecated(self):
        """Deprecated warning for pywbem CIM Objects"""
        msg = _format("Ordering comparisons involving {0} objects are "
                      "deprecated.", self.__class__.__name__)
        if DEBUG_WARNING_ORIGIN:
            msg += "\nTraceback:\n" + ''.join(traceback.format_stack())
        warnings.warn(msg, DeprecationWarning, stacklevel=3)

    def __lt__(self, other):
        """
        Invoked when two CIM objects are compared with the `<` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        self.__ordering_deprecated()
        return self._cmp(other) < 0

    def __gt__(self, other):
        """
        Invoked when two CIM objects are compared with the `>` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        self.__ordering_deprecated()
        return self._cmp(other) > 0

    def __le__(self, other):
        """
        Invoked when two CIM objects are compared with the `<=` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        self.__ordering_deprecated()
        return self._cmp(other) <= 0

    def __ge__(self, other):
        """
        Invoked when two CIM objects are compared with the `>=` operator.

        The comparison is delegated to the `_cmp()` method.
        """
        self.__ordering_deprecated()
        return self._cmp(other) >= 0

    def _cmp(self, other):
        """
        Interface definition for comparator method to be provided by
        subclasses, as follows:
        * If self == other, 0 must be returned.
        * If self < other, -1 must be returned.
        * If self > other, +1 must be returned.
        """
        raise NotImplementedError

    def __hash__(self):
        """
        Interface definition for hash function to be provided by subclasses.

        Background: In order to behave as expected in sets and other hash-based
        collections, the hash values of objects must be equal when the objects
        themselves are considered equal. The default hash function for classes
        is based on `id()` and therefore does not satisfy that requirement.

        Therefore, the CIM objects need to implement a hash function that
        satisfies that requirement.
        """
        raise NotImplementedError


[docs]class MinutesFromUTC(tzinfo): """ Timezone information (an implementation of :class:`py:datetime.tzinfo`) that represents a fixed offset in +/- minutes from UTC and is thus suitable for the CIM datetime data type. Objects of this class are needed in order to make :class:`py:datetime.datetime` objects timezone-aware, in order to be useable as input data to the timezone-aware :class:`~pywbem.CIMDateTime` type. They are also used to provide timezone information to :meth:`~pywbem.CIMDateTime.now` and :meth:`~pywbem.CIMDateTime.fromtimestamp` Example: :: from datetime import datetime from time import time import pywbem # Create a timezone-aware datetime object (for CEDT = UTC+2h), and # convert that to CIM datetime: dt = datetime(year=2016, month=3, day=31, hour=19, minute=30, second=40, microsecond=654321, tzinfo=pywbem.MinutesFromUTC(120)) cim_dt = pywbem.CIMDateTime(dt) # Convert a POSIX timestamp value to CIM datetime (for EST = UTC-5h): posix_ts = time() # seconds since the epoch, not timezone-aware cim_dt = pywbem.CIMDateTime.fromtimestamp(posix_ts, pywbem.MinutesFromUTC(-300)) """ def __init__(self, offset): # pylint: disable=super-init-not-called """ Parameters: offset (:term:`integer`): Timezone offset to be represented in the CIM datetime value in +/- minutes from UTC. This is the offset of local time to UTC (including DST offset), where a positive value indicates minutes east of UTC, and a negative value indicates minutes west of UTC. """ self._offset = offset def __repr__(self): return _format( "MinutesFromUTC(" "offset={s._offset!A})", s=self)
[docs] def utcoffset(self, dt): # pylint: disable=unused-argument """ An implementation of the corresponding base class method (see :meth:`py:datetime.tzinfo.utcoffset` for its description), which needs to return the offset of local time to UTC (including DST offset), as a :class:`py:datetime.timedelta` object. This method is called by the Python datetime classes, and a pywbem user normally does not have to deal with it. This implementation returns the offset used to initialize the object, for any specified `dt` parameter. """ return timedelta(minutes=self._offset)
[docs] def dst(self, dt): # pylint: disable=unused-argument """ An implementation of the corresponding base class method, (see :meth:`py:datetime.tzinfo.dst` for its description), which needs to return the offset caused by DST, as a :class:`py:datetime.timedelta` object. This method is called by the Python datetime classes, and a pywbem user normally does not have to deal with it. This implementation returns an offset of 0 (indicating that DST is not in effect), for any specified `dt` parameter, because CIM datetime values do not represent DST information. """ return timedelta(0)
[docs]class CIMType(object): # pylint: disable=too-few-public-methods """Base type for all CIM data types defined in this package.""" # Note: __str__() is not needed; the inherited method is used, # even though there is a __repr__() method here. #: The name of the CIM datatype, as a :term:`string`. See #: :ref:`CIM data types` for details. cimtype = None
[docs] def __repr__(self): """Return a string representation suitable for debugging.""" return _format( "{s.__class__.__name__}(" "cimtype={s.cimtype!A}, " "{s})", s=self)
[docs]class CIMDateTime(CIMType, _CIMComparisonMixin): """ A value of CIM data type datetime. The object represents either a timezone-aware point in time, or a time interval. Two objects of this class compare equal if their public attributes compare equal. Objects of this class are immutable and :term:`hashable`, with the hash value being based on their public attributes. """ #: The name of the CIM datatype ``"datetime"`` cimtype = 'datetime' _timestamp_pattern = re.compile( r'^([\d\*]{4})([\d\*]{2})([\d\*]{2})' r'([\d\*]{2})([\d\*]{2})([\d\*]{2})\.([\d\*]{6})' r'([+|-])(\d{3})') _interval_pattern = re.compile( r'^([\d\*]{8})([\d\*]{2})([\d\*]{2})([\d\*]{2})\.([\d\*]{6})' r'(:)(000)') def __init__(self, dtarg): """ Parameters: dtarg: The value from which the object is initialized, as one of the following types: * A :term:`string` object will be interpreted as CIM datetime format (see :term:`DSP0004`) and will result in a point in time or a time interval. The use of asterisk characters in the value is supported according to the rules defined in :term:`DSP0004` (e.g. "20180911124613.128***:000"). * A :class:`py:datetime.datetime` object will result in a point in time. If the :class:`py:datetime.datetime` object is timezone-aware (see :class:`~pywbem.MinutesFromUTC`), the specified timezone will be used. Otherwise, a default timezone of UTC will be assumed. * A :class:`py:datetime.timedelta` object will result in a time interval. * Another :class:`~pywbem.CIMDateTime` object will be copied. """ self.__timedelta = None # timedelta value, if interval self.__datetime = None # datetime value, if timestamp self.__precision = None # 0-based index of first asterisk, or None dtarg = _ensure_unicode(dtarg) if isinstance(dtarg, six.text_type): m = self._timestamp_pattern.search(dtarg) if m is not None: # timestamp format parts = m.groups() offset = int(parts[8]) if parts[7] == '-': offset = -offset if '*' in dtarg: first = dtarg.index('*') after = dtarg.rindex('*') + 1 if not re.match(r'^[\*\.]+$', dtarg[first:after]): raise ValueError( _format("Asterisks in CIM datetime timestamp " "value are not consecutive: {0!A}", dtarg)) if after != 21: # end of microseconds field raise ValueError( _format("Asterisks in CIM datetime timestamp " "value do not include least significant " "field: {0!A}", dtarg)) self.__precision = first year = self._to_int(parts[0], 0, None, 'year', dtarg) month = self._to_int(parts[1], 1, None, 'month', dtarg) day = self._to_int(parts[2], 1, None, 'day', dtarg) hour = self._to_int(parts[3], 0, None, 'hour', dtarg) minute = self._to_int(parts[4], 0, None, 'minute', dtarg) second = self._to_int(parts[5], 0, None, 'second', dtarg) microsec = self._to_int(parts[6], 0, '0', 'microsecond', dtarg) try: # Possible errors are e.g. field out of range self.__datetime = datetime( year, month, day, hour, minute, second, microsec, MinutesFromUTC(offset)) except ValueError as exc: raise ValueError( _format("Invalid datetime() input from CIM datetime " "timestamp value {0!A}: {1}", dtarg, exc)) else: m = self._interval_pattern.search(dtarg) if m is not None: # interval format parts = m.groups() if '*' in dtarg: first = dtarg.index('*') after = dtarg.rindex('*') + 1 if not re.match(r'^[\*\.]+$', dtarg[first:after]): raise ValueError( _format("Asterisks in CIM datetime interval " "value are not consecutive: {0!A}", dtarg)) if after != 21: # end of microseconds field raise ValueError( _format("Asterisks in CIM datetime interval " "value do not include least " "significant field: {0!A}", dtarg)) self.__precision = first days = self._to_int(parts[0], 0, None, 'days', dtarg) hours = self._to_int(parts[1], 0, None, 'hours', dtarg) minutes = self._to_int(parts[2], 0, None, 'minutes', dtarg) seconds = self._to_int(parts[3], 0, None, 'seconds', dtarg) microsecs = self._to_int(parts[4], 0, '0', 'microseconds', dtarg) try: # Possible errors are e.g. field out of range self.__timedelta = timedelta( days=days, hours=hours, minutes=minutes, seconds=seconds, microseconds=microsecs) except ValueError as exc: raise ValueError( _format("Invalid timedelta() input from CIM " "datetime interval value {0!A}: {1}", dtarg, exc)) else: raise ValueError( _format("Invalid format of CIM datetime value: {0!A}", dtarg)) elif isinstance(dtarg, datetime): if dtarg.tzinfo is None: self.__datetime = dtarg.replace(tzinfo=MinutesFromUTC(0)) else: self.__datetime = copy.copy(dtarg) elif isinstance(dtarg, timedelta): self.__timedelta = copy.copy(dtarg) elif isinstance(dtarg, CIMDateTime): self.__datetime = copy.copy(dtarg.datetime) self.__timedelta = copy.copy(dtarg.timedelta) else: raise TypeError( _format("dtarg argument {0!A} has an invalid type: {1} " "(expected datetime, timedelta, string, or " "CIMDateTime)", dtarg, type(dtarg))) @staticmethod def _to_int(value_str, min_value, rep_digit, field_name, dtarg): """ Convert value_str into an integer, replacing right-consecutive asterisks with rep_digit, and an all-asterisk value with min_value. field_name and dtarg are passed only for informational purposes. """ if '*' in value_str: first = value_str.index('*') after = value_str.rindex('*') + 1 if value_str[first:after] != '*' * (after - first): raise ValueError( _format("Asterisks in {0} field of CIM datetime value " "{1!A} are not consecutive: {2!A}", field_name, dtarg, value_str)) if after != len(value_str): raise ValueError( _format("Asterisks in {0} field of CIM datetime value " "{1!A} do not end at end of field: {2!A}", field_name, dtarg, value_str)) if rep_digit is None: # pylint: disable=no-else-return # Must be an all-asterisk field if first != 0: raise ValueError( _format("Asterisks in {0} field of CIM datetime value " "{1!A} do not start at begin of field: {2!A}", field_name, dtarg, value_str)) return min_value else: value_str = value_str.replace('*', rep_digit) # Because the pattern and the asterisk replacement mechanism already # ensure only decimal digits, we expect the integer conversion to # always succeed. value = int(value_str) return value def _to_str(self, value, field_begin, field_len): """ Convert value (int) to a field string, considering precision. """ value_str = '{0:0{1}d}'.format(value, field_len) if self.precision is not None and \ self.precision < field_begin + field_len: # field is partly or completely affected by precision # -> replace insignificant digits with asterisks precision_index = max(0, self.precision - field_begin) value_str = value_str[:precision_index] + \ '*' * (field_len - precision_index) return value_str @property def minutes_from_utc(self): """ The timezone offset of this point in time object as +/- minutes from UTC. A positive value of the timezone offset indicates minutes east of UTC, and a negative value indicates minutes west of UTC. 0, if this object represents a time interval. """ offset = 0 if self.__datetime is not None and \ self.__datetime.utcoffset() is not None: offset = self.__datetime.utcoffset().seconds / 60 if self.__datetime.utcoffset().days == -1: offset = -((60 * 24) - offset) return int(offset) @property def datetime(self): """ The point in time represented by this object, as a :class:`py:datetime.datetime` object. `None` if this object represents a time interval. """ return self.__datetime @property def timedelta(self): """ The time interval represented by this object, as a :class:`py:datetime.timedelta` object. `None` if this object represents a point in time. """ return self.__timedelta @property def precision(self): """ Precision of the time interval or point in time represented by this object, if the datetime input string contained asterisk characters. The precision is the 0-based index of the first asterisk character in the datetime input string, or `None` if there were no asterisk characters. For example, the precision of the timestamp value "201809121230**.******+000" is 12. """ return self.__precision @property def is_interval(self): """ A boolean indicating whether this object represents a time interval (`True`) or a point in time (`False`). """ return self.__timedelta is not None
[docs] @staticmethod def get_local_utcoffset(): """ Return the timezone offset of the current local timezone as +/- minutes from UTC. A positive value indicates minutes east of UTC, and a negative value indicates minutes west of UTC. """ utc = datetime.utcnow() local = datetime.now() if local < utc: return - int(float((utc - local).seconds) / 60 + .5) return int(float((local - utc).seconds) / 60 + .5)
[docs] @classmethod def now(cls, tzi=None): """ Factory method that returns a new :class:`~pywbem.CIMDateTime` object representing the current date and time. The optional timezone information is used to convert the CIM datetime value into the desired timezone. That does not change the point in time that is represented by the value, but it changes the value of the ``hhmmss`` components of the CIM datetime value to compensate for changes in the timezone offset component. Parameters: tzi (:class:`~pywbem.MinutesFromUTC`): Timezone information. `None` means that the current local timezone is used. Returns: A new :class:`~pywbem.CIMDateTime` object representing the current date and time. """ if tzi is None: tzi = MinutesFromUTC(cls.get_local_utcoffset()) return cls(datetime.now(tzi))
[docs] @classmethod def fromtimestamp(cls, ts, tzi=None): # pylint: disable=invalid-name """ Factory method that returns a new :class:`~pywbem.CIMDateTime` object from a POSIX timestamp value and optional timezone information. A POSIX timestamp value is the number of seconds since "the epoch", i.e. 1970-01-01 00:00:00 UTC. Thus, a POSIX timestamp value is unambiguous w.r.t. the timezone, but it is not timezone-aware. The optional timezone information is used to convert the CIM datetime value into the desired timezone. That does not change the point in time that is represented by the value, but it changes the value of the ``hhmmss`` components of the CIM datetime value to compensate for changes in the timezone offset component. Parameters: ts (:term:`integer`): POSIX timestamp value. tzi (:class:`~pywbem.MinutesFromUTC`): Timezone information. `None` means that the current local timezone is used. Returns: A new :class:`~pywbem.CIMDateTime` object representing the specified point in time. """ if tzi is None: tzi = MinutesFromUTC(cls.get_local_utcoffset()) return cls(datetime.fromtimestamp(ts, tzi))
[docs] def __str__(self): """ Return a string representing the object in CIM datetime format. """ if self.is_interval: # pylint: disable=no-else-return days = self.timedelta.days hours = self.timedelta.seconds // 3600 sec_in_hour = self.timedelta.seconds - hours * 3600 minutes = sec_in_hour // 60 seconds = sec_in_hour - minutes * 60 microsecs = self.timedelta.microseconds days_str = self._to_str(days, 0, 8) hours_str = self._to_str(hours, 8, 2) minutes_str = self._to_str(minutes, 10, 2) seconds_str = self._to_str(seconds, 12, 2) microsecs_str = self._to_str(microsecs, 15, 6) ret_str = '{0}{1}{2}{3}.{4}:000'.format( days_str, hours_str, minutes_str, seconds_str, microsecs_str) return ret_str else: # timestamp offset = self.minutes_from_utc sign = '+' if offset < 0: sign = '-' offset = -offset year = self.datetime.year month = self.datetime.month day = self.datetime.day hour = self.datetime.hour minute = self.datetime.minute second = self.datetime.second microsec = self.datetime.microsecond year_str = self._to_str(year, 0, 4) month_str = self._to_str(month, 4, 2) day_str = self._to_str(day, 6, 2) hour_str = self._to_str(hour, 8, 2) minute_str = self._to_str(minute, 10, 2) second_str = self._to_str(second, 12, 2) microsec_str = self._to_str(microsec, 15, 6) ret_str = '{0}{1}{2}{3}{4}{5}.{6}{7}{8:03d}'.format( year_str, month_str, day_str, hour_str, minute_str, second_str, microsec_str, sign, offset) return ret_str
[docs] def __repr__(self): """ Return a string representation suitable for debugging. """ return _format( "CIMDateTime(" "cimtype={s.cimtype!A}, " "datetime={s.datetime!A}, " "timedelta={s.timedelta!A}, " "precision={s.precision!A}), " "minutes_from_utc={s.minutes_from_utc!A})", s=self)
def __getstate__(self): return str(self) def __setstate__(self, arg): self.__init__(arg) def _cmp(self, other): # Defer import due to circular import dependencies: from .cim_obj import cmpitem if self is other: return 0 if not isinstance(other, CIMDateTime): return 1 return (cmpitem(self.datetime, other.datetime) or cmpitem(self.timedelta, other.timedelta)) def __hash__(self): """ Return a hash value based on the public attributes of this class. Because these attributes are not modifiable, objects of this class are :term:`hashable` (and not just :term:`unchanged-hashable`). """ hashes = ( _hash_item(self.datetime), _hash_item(self.timedelta), # The 'is_interval' and 'minutes_from_utc' attributes are not used # for hash value calculation because they are derived attributes. ) return hash(hashes)
# CIM integer types
[docs]class CIMInt(CIMType, _Longint): """ Base type for CIM integer data types. Derived from :class:`~pywbem.CIMType` and :class:`py:int` (for Python 3) or :class:`py:long` (for Python 2). This class has a concept of a valid range for the represented integer, based upon the capability of the CIM data type as defined in :term:`DSP0004`. The additional constraints defined by possible MinValue or MaxValue qualifiers are not taken into account at this level. The valid value range is enforced when an instance of a subclass of this class (e.g. :class:`~pywbem.Uint8`) is created. Values outside of the valid range raise a :exc:`ValueError`. The enforcement of the valid value range can be disabled via the configuration variable :data:`~pywbem.config.ENFORCE_INTEGER_RANGE`. Two objects of subclasses of this base class compare equal if their numeric values compare equal. Objects of this class are immutable and :term:`hashable`, with the hash value being based on its numeric value. Instances of subclasses of this class can be initialized with the usual input arguments supported by :term:`integer`, for example: :: >>> pywbem.Uint8(42) Uint8(cimtype='uint8', 42) >>> pywbem.Uint8('42') Uint8(cimtype='uint8', 42) >>> pywbem.Uint8('2A', 16) Uint8(cimtype='uint8', 42) >>> pywbem.Uint8('100', 16) Traceback (most recent call last): . . . ValueError: Integer value 256 is out of range for CIM datatype uint8 >>> pywbem.Uint8(100, 10) Traceback (most recent call last): . . . TypeError: int() can't convert non-string with explicit base """ #: The minimum valid value for the integer, according to the capabilities #: of its CIM data type. See :ref:`CIM data types` for a list of CIM #: integer data types. minvalue = None #: The maximum valid value for the integer, according to the capabilities #: of its CIM data type. See :ref:`CIM data types` for a list of CIM #: integer data types. maxvalue = None def __new__(cls, *args, **kwargs): # Python 3.7 removed support for passing the value for int() as a # keyword argument named 'x'. It now needs to be passed as a positional # argument. The testclient test case definitions rely on a keyword # argument, so we now transform the keyword arg into a positional # arg. if 'x' in kwargs: args = list(*args) # args is passed as a tuple args.append(kwargs.pop('x')) value = _Longint(*args, **kwargs) if ENFORCE_INTEGER_RANGE: if value > cls.maxvalue or value < cls.minvalue: raise ValueError( _format("Integer value {0} is out of range for CIM " "datatype {1}", value, cls.cimtype)) # The value needs to be processed here, because int/long is unmutable return super(CIMInt, cls).__new__(cls, *args, **kwargs)
[docs] def __repr__(self): """ Return a string representation suitable for debugging. """ return _format( "{s.__class__.__name__}(" "cimtype={s.cimtype!A}, " "minvalue={s.minvalue}, " # Avoid long indicator 'L' in Python 2 "maxvalue={s.maxvalue}, " # Avoid long indicator 'L' in Python 2 "{s})", s=self)
[docs]class Uint8(CIMInt): """ A value of CIM data type uint8. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'uint8' #: The minimum valid value for the CIM datatype minvalue = 0 #: The maximum valid value for the CIM datatype maxvalue = 2**8 - 1
[docs]class Sint8(CIMInt): """ A value of CIM data type sint8. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'sint8' #: The minimum valid value for the CIM datatype minvalue = -2 ** (8 - 1) #: The maximum valid value for the CIM datatype maxvalue = 2 ** (8 - 1) - 1
[docs]class Uint16(CIMInt): """ A value of CIM data type uint16. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'uint16' #: The minimum valid value for the CIM datatype minvalue = 0 #: The maximum valid value for the CIM datatype maxvalue = 2**16 - 1
[docs]class Sint16(CIMInt): """ A value of CIM data type sint16. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'sint16' #: The minimum valid value for the CIM datatype minvalue = -2 ** (16 - 1) #: The maximum valid value for the CIM datatype maxvalue = 2 ** (16 - 1) - 1
[docs]class Uint32(CIMInt): """ A value of CIM data type uint32. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'uint32' #: The minimum valid value for the CIM datatype minvalue = 0 #: The maximum valid value for the CIM datatype maxvalue = 2 ** 32 - 1
[docs]class Sint32(CIMInt): """ A value of CIM data type sint32. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'sint32' #: The minimum valid value for the CIM datatype minvalue = -2 ** (32 - 1) #: The maximum valid value for the CIM datatype maxvalue = 2 ** (32 - 1) - 1
[docs]class Uint64(CIMInt): """ A value of CIM data type uint64. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'uint64' #: The minimum valid value for the CIM datatype minvalue = 0 #: The maximum valid value for the CIM datatype maxvalue = 2 ** 64 - 1
[docs]class Sint64(CIMInt): """ A value of CIM data type sint64. Derived from :class:`~pywbem.CIMInt`. For details on CIM integer data types, see :class:`~pywbem.CIMInt`. """ #: The name of the CIM datatype cimtype = 'sint64' #: The minimum valid value for the CIM datatype minvalue = -2 ** (64 - 1) #: The maximum valid value for the CIM datatype maxvalue = 2 ** (64 - 1) - 1
# CIM float types
[docs]class CIMFloat(CIMType, float): """ Base type for real (floating point) CIM data types. Two objects of subclasses of this base class compare equal if their numeric values compare equal. Objects of this class are immutable and :term:`hashable`, with the hash value being based on its numeric value. Note that equality comparison of floating point numbers in Python (and in almost any programming language) comes with some surprises. See `"Floating Point Arithmetic: Issues and Limitations" <https://docs.python.org/2/tutorial/floatingpoint.html>`_ for details, and specifically `"Comparing Floating Point Numbers, 2012 Edition" <https://randomascii.wordpress.com/2012/02/25/comparing-floating-point-numbers-2012-edition/>`_ on the topic of equality comparison. The same issues apply to hash values that are based on the numeric value of floating point numbers. Therefore, it is not recommended to perform equality comparison of objects of subclasses of this class, or to use them as dictionary keys or as members in sets. """
[docs]class Real32(CIMFloat): """ A value of CIM data type real32. Derived from :class:`~pywbem.CIMFloat`. It is not recommended to perform equality comparison on objects of this class, or to use them as dictionary keys or as members in sets. See :class:`~pywbem.CIMFloat` for details. """ #: The name of the CIM datatype cimtype = 'real32'
[docs]class Real64(CIMFloat): """ A value of CIM data type real64. Derived from :class:`~pywbem.CIMFloat`. It is not recommended to perform equality comparison on objects of this class, or to use them as dictionary keys or as members in sets. See :class:`~pywbem.CIMFloat` for details. """ #: The name of the CIM datatype cimtype = 'real64'
# Python number types listed in :term:`number`. number_types = six.integer_types + (float,) # pylint: disable=invalid-name
[docs]def cimtype(obj): """ Return the CIM data type name of a CIM typed object, as a string. For an array, the type is determined from the first array element (CIM arrays must be homogeneous w.r.t. the type of their elements). If the array is empty, that is not possible and :exc:`~py:exceptions.ValueError` is raised. Note that Python :term:`numbers <number>` are not valid input objects because determining their CIM data type (e.g. :class:`~pywbem.Uint8`, :class:`~pywbem.Real32`) would require knowing the value range. Therefore, :exc:`~py:exceptions.TypeError` is raised in this case. Parameters: obj (:term:`CIM data type`): The object whose CIM data type name is returned. Returns: :term:`string`: The CIM data type name of the object (e.g. ``"uint8"``). Raises: TypeError: The object does not have a valid CIM data type. ValueError: Cannot determine CIM data type from an empty array. """ if isinstance(obj, CIMType): return obj.cimtype if isinstance(obj, bool): return 'boolean' if isinstance(obj, (six.binary_type, six.text_type)): # accept both possible types return 'string' if isinstance(obj, list): try: obj = obj[0] except IndexError: raise ValueError("Cannot determine CIM data type from empty array") return cimtype(obj) if isinstance(obj, (datetime, timedelta)): return 'datetime' try: instancename_type = CIMInstanceName except NameError: # Defer import due to circular import dependencies: from pywbem.cim_obj import CIMInstanceName as instancename_type if isinstance(obj, instancename_type): return 'reference' try: instance_type = CIMInstance except NameError: # Defer import due to circular import dependencies: from pywbem.cim_obj import CIMInstance as instance_type if isinstance(obj, instance_type): # embedded instance return 'string' try: class_type = CIMClass except NameError: # Defer import due to circular import dependencies: from pywbem.cim_obj import CIMClass as class_type if isinstance(obj, class_type): # embedded class return 'string' raise TypeError( _format("Object does not have a valid CIM data type: {0!A}", obj))
_TYPE_FROM_NAME = { 'boolean': bool, 'string': six.text_type, # return the preferred type 'char16': six.text_type, # return the preferred type 'datetime': CIMDateTime, # 'reference' covered at run time 'uint8': Uint8, 'uint16': Uint16, 'uint32': Uint32, 'uint64': Uint64, 'sint8': Sint8, 'sint16': Sint16, 'sint32': Sint32, 'sint64': Sint64, 'real32': Real32, 'real64': Real64, }
[docs]def type_from_name(type_name): """ Return the Python type object for a given CIM data type name. For example, type name ``"uint8"`` will return type object :class:`~pywbem.Uint8`. For CIM data type names ``"string"`` and ``"char16"``, the :term:`unicode string` type is returned (Unicode strings are the preferred representation for these CIM data types). The returned type can be used as a constructor from a differently typed input value in many cases. Notable limitations are: * In Python 3, the :class:`py3:str` type is used to represent CIM string data types. When constructing such an object from a byte string, the resulting string is not a unicode-translated version of the byte string as one would assume (and as is the case in Python 2), but instead that results in a unicode string that is a `repr()` representation of the byte string:: string_type = type_from_name('string') # str s1 = b'abc' s2 = string_type(s1) # results in u"b'abc'", and not in u"abc" Use `decode()` and `encode()` for strings instead of type conversion syntax (in both Python 2 and 3, for consistency). Parameters: type_name (:term:`string`): The simple (=non-array) CIM data type name (e.g. ``"uint8"`` or ``"reference"``). Returns: The Python type object for the CIM data type (e.g. :class:`~pywbem.Uint8` or :class:`~pywbem.CIMInstanceName`). Raises: ValueError: Unknown CIM data type name. """ if type_name == 'reference': # Defer import due to circular import dependencies: from .cim_obj import CIMInstanceName return CIMInstanceName try: type_obj = _TYPE_FROM_NAME[type_name] except KeyError: raise ValueError( _format("Unknown CIM data type name: {0!A}", type_name)) return type_obj
def atomic_to_cim_xml(obj): """ Convert an "atomic" scalar value to a CIM-XML string and return that string. The returned CIM-XML string is ready for use as the text of a CIM-XML 'VALUE' element. Parameters: obj (:term:`CIM data type`, :term:`number`, :class:`py:datetime`): The "atomic" input value. May be `None`. Must not be an array/list/tuple. Must not be a :ref:`CIM object`. Returns: A :term:`unicode string` object in CIM-XML value format representing the input value. `None`, if the input value is `None`. Raises: TypeError """ if obj is None: # pylint: disable=no-else-return return obj elif isinstance(obj, six.text_type): return obj elif isinstance(obj, six.binary_type): return _to_unicode(obj) elif isinstance(obj, bool): return u'TRUE' if obj else u'FALSE' elif isinstance(obj, (CIMInt, six.integer_types, CIMDateTime)): return six.text_type(obj) elif isinstance(obj, datetime): return six.text_type(CIMDateTime(obj)) elif isinstance(obj, Real32): # DSP0201 requirements for representing real32: # The significand must be represented with at least 11 digits. # The special values must have the case: INF, -INF, NaN. s = u'{0:.11G}'.format(obj) if s == 'NAN': s = u'NaN' elif s in ('INF', '-INF'): pass elif '.' not in s: parts = s.split('E') parts[0] = parts[0] + '.0' s = 'E'.join(parts) return s elif isinstance(obj, (Real64, float)): # DSP0201 requirements for representing real64: # The significand must be represented with at least 17 digits. # The special values must have the case: INF, -INF, NaN. s = u'{0:.17G}'.format(obj) if s == 'NAN': s = u'NaN' elif s in ('INF', '-INF'): pass elif '.' not in s: parts = s.split('E') parts[0] = parts[0] + '.0' s = 'E'.join(parts) return s else: raise TypeError( _format("Value {0!A} has invalid type {1} for conversion to a " "CIM-XML string", obj, type(obj)))