# Copyright (C) 2012-2013 Peter Hatina <phatina@redhat.com>
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, see <http://www.gnu.org/licenses/>.
from functools import wraps
from LMIUtil import lmi_raise_or_dump_exception
from LMIExceptions import LMIDeletedObjectError
[docs]class lmi_return_expr_if_fail(object):
"""
Decorator, which calls a specified expression and returns its return value instead of
calling the decorated method, if provided test expression is False; otherwise a method
is called.
:param expr_test: expression which determines, if to execute a return
value expression
:param expr_ret: expression, which is called, if the ``expr_test`` returns False
:param expr_ret_args: ``expr_ret`` position arguments
:param expr_ret_kwargs: ``expr_ret`` keyword arguments
:param bool Self: flag, which specifies, if to pass ``self`` variable to the ``expr_ret``,
if ``expr_test`` failed
Example of usage:
.. code-block:: python
class Foo:
def __init__(self, member):
self._member = member
def failed(self):
print "expression failed"
return False
# NOTE: the self parameter to the method call needs to be passed via expr_ret_args,
# therefore, there is a dummy lambda obj: obj, which is basically self variable.
@lmi_return_expr_if_fail(lambda obj: obj._member, failed, lambda obj: obj)
def some_method(self):
print "some_method called"
return True
f = Foo(None)
f.some_method() == False
f = Foo(True)
f.some_method() == True
"""
def __init__(self, expr_test, expr_ret, Self=False, *expr_ret_args, **expr_ret_kwargs):
self._expr_test = expr_test
self._expr_ret = expr_ret
self._expr_ret_pass_self = Self
self._expr_ret_args = expr_ret_args
self._expr_ret_kwargs = expr_ret_kwargs
def __call__(self, fn):
"""
Executes a method call, if the test passed.
:param instancemethod fn: decorated method
"""
@wraps(fn)
def wrapper(self_wr, *args, **kwargs):
failed = False
try:
if not self._expr_test(self_wr):
failed = True
except AttributeError, e:
failed = True
if failed:
if self._expr_ret_pass_self:
return self._expr_ret(self_wr, *self._expr_ret_args, **self._expr_ret_kwargs)
return self._expr_ret(*self._expr_ret_args, **self._expr_ret_kwargs)
return fn(self_wr, *args, **kwargs)
return wrapper
[docs]class lmi_return_val_if_fail(lmi_return_expr_if_fail):
"""
Decorator, which returns a specified value and no method call is performed, if
provided test expression is False; otherwise a method is called.
:param expr_test: if the expression returns False, a method call is called
:param rval: return value of the method, if the object attribute as expression
failed
Example of usage:
.. code-block:: python
class Foo:
def __init__(self, member):
self._member = member
@lmi_return_val_if_fail(lambda obj: obj._member, False)
def some_method(self):
print "some_method called"
return True
f = Foo(None)
f.some_method() == False
f = Foo(True)
f.some_method() == True
"""
def __init__(self, expr_test, rval):
lmi_return_expr_if_fail.__init__(self, expr_test, lambda: rval)
[docs]class lmi_return_if_fail(lmi_return_val_if_fail):
"""
Decorator, which returns None and no method call is performed, if provided test
expression is False; otherwise a method is called.
:param expr_test: if the expression ``expr_test`` returns True, a method is called
Example of usage:
.. code-block:: python
class Foo:
def __init__(self, member):
self._member = member
@lmi_return_if_fail(lambda obj: obj._member)
def some_method(self):
print "some_method called"
return True
f = Foo(None)
f.some_method() == None
f = Foo(True)
f.some_method() == True
"""
def __init__(self, expr_test):
lmi_return_val_if_fail.__init__(self, expr_test, None)
[docs]class lmi_possibly_deleted(lmi_return_expr_if_fail):
"""
Decorator, which returns None, if provided test expression is True.
:param expr_ret: callable or return value used, if ``expr_test`` fails
:param expr_ret_args: expr_ret position arguments
:param expr_ret_kwargs: expr_ret keyword arguments
:param bool Self: flag, which specifies, if to pass ``self`` variable to the
``expr_ret``, if ``expr_test`` failed
Example of usage:
.. code-block:: python
class Foo:
def __init__(self, deleted):
self._deleted = deleted
@lmi_possibly_deleted(lambda obj: obj._member, lambda: False)
def some_method(self):
print "some_method called"
return True
f = Foo(None)
f.some_method() == False
f = Foo(True)
f.some_method() == True
"""
def __init__(self, expr_ret, Self=False, *expr_ret_args, **expr_ret_kwargs):
call_expr = expr_ret if callable(expr_ret) else lambda: expr_ret
lmi_return_expr_if_fail.__init__(self, lambda obj: not obj._deleted,
call_expr, Self=Self, *expr_ret_args, **expr_ret_kwargs)
def __call__(self, fn):
"""
Executes a method call, if the test passed. If the test expression returns False,
specified return value in the constructor is returned. If the LMIShell does not
dump exceptions, an appropriate exception is raised.
:param instancemethod fn: decorated method
:raises: :py:exc:`.LMIDeletedObjectError`
"""
@wraps(fn)
def wrapper(self_wr, *args, **kwargs):
failed = False
try:
if not self._expr_test(self_wr):
failed = True
except AttributeError, e:
failed = True
if failed:
errorstr = "This instance has been deleted from a CIM broker"
lmi_raise_or_dump_exception(LMIDeletedObjectError(errorstr))
if self._expr_ret_pass_self:
return self._expr_ret(self_wr, *self._expr_ret_args, **self._expr_ret_kwargs)
return self._expr_ret(*self._expr_ret_args, **self._expr_ret_kwargs)
return fn(self_wr, *args, **kwargs)
return wrapper
[docs]def lmi_class_fetch_lazy(fn):
"""
Decorator for :py:class:`LMIClass`, which first fetches a wrapped :py:class:`CIMClass`
object and then executes a wrapped method.
:param instancemethod fn: wrapped method
"""
@wraps(fn)
def wrapped(self, *args, **kwargs):
if not self._cim_class:
self.fetch()
return fn(self, *args, **kwargs)
return wrapped