#
# (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 constructors 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 constructor
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
if six.PY2:
_Longint = long # noqa: F821
else:
_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 = "Ordering comparisons involving %s 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 = timedelta(minutes=offset)
def __repr__(self):
return '%s(offset=%r)' % (self.__class__.__name__, self.__offset)
[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 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 '%s(cimtype=%r, %s)' % \
(self.__class__.__name__, self.cimtype, 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.
"""
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.
"""
from .cim_obj import _ensure_unicode # defer due to cyclic deps.
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(
'Asterisks in CIM datetime timestamp value are '
'not consecutive: "%s"' % dtarg)
if after != 21: # end of microseconds field
raise ValueError(
'Asterisks in CIM datetime timestamp value do not '
'include least significant field: "%s"' % 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(
'Invalid datetime() input from CIM datetime timestamp '
'value "%s": %s' % (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(
'Asterisks in CIM datetime interval value are '
'not consecutive: "%s"' % dtarg)
if after != 21: # end of microseconds field
raise ValueError(
'Asterisks in CIM datetime interval value do '
'not include least significant field: "%s"' %
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(
'Invalid timedelta() input from CIM datetime '
'interval value "%s": %s' % (dtarg, exc))
else:
raise ValueError(
'Invalid format of CIM datetime value: "%s"' % 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('dtarg argument "%s" has an invalid type: %s '
'(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(
'Asterisks in %s field of CIM datetime value "%s" '
'are not consecutive: %r' %
(field_name, dtarg, value_str))
if after != len(value_str):
raise ValueError(
'Asterisks in %s field of CIM datetime value "%s" '
'do not end at end of field: %r' %
(field_name, dtarg, value_str))
if rep_digit is None:
# Must be an all-asterisk field
if first != 0:
raise ValueError(
'Asterisks in %s field of CIM datetime value "%s" '
'do not start at begin of field: %r' %
(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 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:
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 = '%s%s%s%s.%s:000' % \
(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 = '%s%s%s%s%s%s.%s%s%03d' % \
(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 '%s(cimtype=%r, datetime=%r, timedelta=%r, precision=%r)' % \
(self.__class__.__name__, self.cimtype, self.datetime,
self.timedelta, self.precision)
def __getstate__(self):
return str(self)
def __setstate__(self, arg):
self.__init__(arg)
def _cmp(self, other):
from .cim_obj import cmpitem # defer due to cyclic deps.
if self is other:
return 0
elif 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`).
"""
from .cim_obj import _hash_item # defer due to cyclic deps.
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):
value = _Longint(*args, **kwargs)
if ENFORCE_INTEGER_RANGE:
if value > cls.maxvalue or value < cls.minvalue:
raise ValueError("Integer value %s is out of range for CIM "
"datatype %s" % (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 '%s(cimtype=%r, minvalue=%s, maxvalue=%s, %s)' % \
(self.__class__.__name__, self.cimtype, self.minvalue,
self.maxvalue, 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`.
"""
cimtype = 'uint8'
minvalue = 0
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`.
"""
cimtype = 'sint8'
minvalue = -2 ** (8 - 1)
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`.
"""
cimtype = 'uint16'
minvalue = 0
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`.
"""
cimtype = 'sint16'
minvalue = -2 ** (16 - 1)
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`.
"""
cimtype = 'uint32'
minvalue = 0
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`.
"""
cimtype = 'sint32'
minvalue = -2 ** (32 - 1)
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`.
"""
cimtype = 'uint64'
minvalue = 0
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`.
"""
cimtype = 'sint64'
minvalue = -2 ** (64 - 1)
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.
"""
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.
"""
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:
from pywbem.cim_obj import CIMInstanceName as instancename_type
if isinstance(obj, instancename_type):
return 'reference'
try:
instance_type = CIMInstance
except NameError:
from pywbem.cim_obj import CIMInstance as instance_type
if isinstance(obj, instance_type): # embedded instance
return 'string'
try:
class_type = CIMClass
except NameError:
from pywbem.cim_obj import CIMClass as class_type
if isinstance(obj, class_type): # embedded class
return 'string'
raise TypeError("Object does not have a valid CIM data type: %r" % 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':
# move import to run time to avoid circular imports
from .cim_obj import CIMInstanceName
return CIMInstanceName
try:
type_obj = _TYPE_FROM_NAME[type_name]
except KeyError:
raise ValueError("Unknown CIM data type name: %r" % 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
"""
from .cim_obj import _ensure_unicode # due to cycles
if obj is None:
return obj
elif isinstance(obj, six.string_types):
return _ensure_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'%.11G' % 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'%.17G' % 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("Value %r has invalid type %s for conversion to a "
"CIM-XML string" % (obj, type(obj)))