diff options
author | Jason Gerard DeRose <jderose@redhat.com> | 2009-01-08 00:07:18 -0700 |
---|---|---|
committer | Jason Gerard DeRose <jderose@redhat.com> | 2009-01-08 00:07:18 -0700 |
commit | 3e9eb0bda000ef138ff04c677aa9014186f547d3 (patch) | |
tree | f88172dadc42d2ed531441eb67df48b64637a640 | |
parent | 166b3ca80c2ed651eb2a5c20bcda5b51ed9e5e2e (diff) | |
download | freeipa-3e9eb0bda000ef138ff04c677aa9014186f547d3.tar.gz freeipa-3e9eb0bda000ef138ff04c677aa9014186f547d3.tar.xz freeipa-3e9eb0bda000ef138ff04c677aa9014186f547d3.zip |
Changed PublicError so str(e) is untranslated (for logging) and added format=None kwarg for generic use
-rw-r--r-- | ipalib/errors2.py | 80 | ||||
-rw-r--r-- | tests/test_ipalib/test_error2.py | 129 |
2 files changed, 156 insertions, 53 deletions
diff --git a/ipalib/errors2.py b/ipalib/errors2.py index 660bb1bd..b052882d 100644 --- a/ipalib/errors2.py +++ b/ipalib/errors2.py @@ -209,31 +209,53 @@ class PluginMissingOverrideError(PrivateError): ############################################################################## # Public errors: + +__messages = [] + +def _(message): + __messages.append(message) + return message + + class PublicError(StandardError): """ **900** Base class for exceptions that can be forwarded in an RPC response. """ errno = 900 - - def __init__(self, message=None, **kw): - if message is None: - message = self.get_format(ugettext) % kw - assert type(message) is unicode - elif type(message) is not unicode: - raise TypeError( - TYPE_ERROR % ('message', unicode, message, type(message)) + format = None + + def __init__(self, format=None, message=None, **kw): + name = self.__class__.__name__ + if self.format is not None and format is not None: + raise ValueError( + 'non-generic %r needs format=None; got format=%r' % ( + name, format) ) - self.message = message + if message is None: + if self.format is None: + if format is None: + raise ValueError( + '%s.format is None yet format=None, message=None' % name + ) + self.format = format + self.forwarded = False + self.message = self.format % kw + self.strerror = ugettext(self.format) % kw + else: + if type(message) is not unicode: + raise TypeError( + TYPE_ERROR % ('message', unicode, message, type(message)) + ) + self.forwarded = True + self.message = message + self.strerror = message for (key, value) in kw.iteritems(): assert not hasattr(self, key), 'conflicting kwarg %s.%s = %r' % ( - self.__class__.__name__, key, value, + name, key, value, ) setattr(self, key, value) - StandardError.__init__(self, message) - - def get_format(self, _): - return _('') + StandardError.__init__(self, self.message) class VersionError(PublicError): @@ -250,11 +272,8 @@ class VersionError(PublicError): """ errno = 901 + format = _('%(cver)s client incompatible with %(sver)s server at %(server)r') - def get_format(self, _): - return _( - '%(cver)s client incompatible with %(sver)s server at %(server)r' - ) class InternalError(PublicError): @@ -270,15 +289,13 @@ class InternalError(PublicError): """ errno = 902 + format = _('an internal error has occured') def __init__(self, message=None): """ Security issue: ignore any information given to constructor. """ - PublicError.__init__(self, self.get_format(ugettext)) - - def get_format(self, _): - return _('an internal error has occured') + PublicError.__init__(self) class ServerInternalError(PublicError): @@ -294,9 +311,7 @@ class ServerInternalError(PublicError): """ errno = 903 - - def get_format(self, _): - return _('an internal error has occured on server at %(server)r') + format = _('an internal error has occured on server at %(server)r') class CommandError(PublicError): @@ -312,9 +327,7 @@ class CommandError(PublicError): """ errno = 904 - - def get_format(self, _): - return _('unknown command %(name)r') + format = _('unknown command %(name)r') class ServerCommandError(PublicError): @@ -331,9 +344,7 @@ class ServerCommandError(PublicError): """ errno = 905 - - def get_format(self, _): - return _('error on server %(server)r: %(error)s') + format = _('error on server %(server)r: %(error)s') class NetworkError(PublicError): @@ -349,9 +360,7 @@ class NetworkError(PublicError): """ errno = 906 - - def get_format(self, _): - return _('cannot connect to %(uri)r') + format = _('cannot connect to %(uri)r') class ServerNetworkError(PublicError): @@ -368,9 +377,8 @@ class ServerNetworkError(PublicError): """ errno = 907 + format = _('error on server %(server)r: %(error)s') - def get_format(self, _): - return _('error on server %(server)r: %(error)s') ############################################################################## diff --git a/tests/test_ipalib/test_error2.py b/tests/test_ipalib/test_error2.py index 85bb0c2b..cd13ba77 100644 --- a/tests/test_ipalib/test_error2.py +++ b/tests/test_ipalib/test_error2.py @@ -24,7 +24,7 @@ Test the `ipalib.error2` module. import re import inspect from tests.util import assert_equal, raises, dummy_ugettext -from ipalib import errors2 +from ipalib import errors2, request from ipalib.constants import TYPE_ERROR @@ -201,23 +201,21 @@ class PublicExceptionTester(object): return self.__klass klass = property(__get_klass) - def new(self, message=None, **kw): + def new(self, format=None, message=None, **kw): # Test that TypeError is raised if message isn't unicode: - e = raises(TypeError, self.klass, 'The message') + e = raises(TypeError, self.klass, message='The message') assert str(e) == TYPE_ERROR % ('message', unicode, 'The message', str) # Test the instance: for (key, value) in kw.iteritems(): assert not hasattr(self.klass, key), key - inst = self.klass(message=message, **kw) + inst = self.klass(format=format, message=message, **kw) assert isinstance(inst, StandardError) assert isinstance(inst, errors2.PublicError) assert isinstance(inst, self.klass) assert not isinstance(inst, errors2.PrivateError) for (key, value) in kw.iteritems(): assert getattr(inst, key) is value - assert str(inst) == inst.get_format(lambda m: m) % kw - assert inst.message == str(inst) return inst @@ -231,35 +229,127 @@ class test_PublicError(PublicExceptionTester): """ Test the `ipalib.errors2.PublicError.__init__` method. """ - inst = self.klass(key1='Value 1', key2='Value 2') - assert inst.key1 == 'Value 1' - assert inst.key2 == 'Value 2' - assert str(inst) == '' + context = request.context + message = u'The translated, interpolated message' + format = 'key=%(key1)r and key2=%(key2)r' + uformat = u'Translated key=%(key1)r and key2=%(key2)r' + val1 = 'Value 1' + val2 = 'Value 2' + kw = dict(key1=val1, key2=val2) + + assert not hasattr(context, 'ugettext') + + # Test with format=str, message=None + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + inst = self.klass(format, **kw) + assert dummy.message is format # Means ugettext() called + assert inst.format is format + assert_equal(inst.message, format % kw) + assert_equal(inst.strerror, uformat % kw) + assert inst.forwarded is False + assert inst.key1 is val1 + assert inst.key2 is val2 + + # Test with format=None, message=unicode + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + inst = self.klass(message=message, **kw) + assert not hasattr(dummy, 'message') # Means ugettext() not called + assert inst.format is None + assert inst.message is message + assert inst.strerror is message + assert inst.forwarded is True + assert inst.key1 is val1 + assert inst.key2 is val2 + + # Test with format=None, message=str + e = raises(TypeError, self.klass, message='the message', **kw) + assert str(e) == TYPE_ERROR % ('message', unicode, 'the message', str) + + # Test with format=None, message=None + e = raises(ValueError, self.klass, **kw) + assert str(e) == \ + 'PublicError.format is None yet format=None, message=None' + + + ###################################### + # Test via PublicExceptionTester.new() - # Test subclass and use of message, get_format(): + # Test with format=str, message=None + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + inst = self.new(format, **kw) + assert isinstance(inst, self.klass) + assert dummy.message is format # Means ugettext() called + assert inst.format is format + assert_equal(inst.message, format % kw) + assert_equal(inst.strerror, uformat % kw) + assert inst.forwarded is False + assert inst.key1 is val1 + assert inst.key2 is val2 + + # Test with format=None, message=unicode + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + inst = self.new(message=message, **kw) + assert isinstance(inst, self.klass) + assert not hasattr(dummy, 'message') # Means ugettext() not called + assert inst.format is None + assert inst.message is message + assert inst.strerror is message + assert inst.forwarded is True + assert inst.key1 is val1 + assert inst.key2 is val2 + + + ################## + # Test a subclass: class subclass(self.klass): - def get_format(self, _): - return _('%(true)r %(text)r %(number)r') + format = '%(true)r %(text)r %(number)r' + uformat = u'Translated %(true)r %(text)r %(number)r' kw = dict(true=True, text='Hello!', number=18) + + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + + # Test with format=str, message=None + e = raises(ValueError, subclass, format, **kw) + assert str(e) == 'non-generic %r needs format=None; got format=%r' % ( + 'subclass', format) + + # Test with format=None, message=None: inst = subclass(**kw) + assert dummy.message is subclass.format # Means ugettext() called + assert inst.format is subclass.format + assert_equal(inst.message, subclass.format % kw) + assert_equal(inst.strerror, uformat % kw) + assert inst.forwarded is False assert inst.true is True assert inst.text is kw['text'] assert inst.number is kw['number'] - assert_equal(inst.message, u'%(true)r %(text)r %(number)r' % kw) - # Test via PublicExceptionTester.new() - inst = self.new(**kw) - assert isinstance(inst, self.klass) + # Test with format=None, message=unicode: + dummy = dummy_ugettext(uformat) + context.ugettext = dummy + inst = subclass(message=message, **kw) + assert not hasattr(dummy, 'message') # Means ugettext() not called + assert inst.format is subclass.format + assert inst.message is message + assert inst.strerror is message + assert inst.forwarded is True assert inst.true is True assert inst.text is kw['text'] assert inst.number is kw['number'] + del context.ugettext def test_public_errors(): """ Test the `ipalib.errors2.public_errors` module variable. """ + i = 0 for klass in errors2.public_errors: assert issubclass(klass, StandardError) assert issubclass(klass, errors2.PublicError) @@ -274,3 +364,8 @@ def test_public_errors(): assert errno == klass.errno, ( 'docstring=%r but errno=%r in %s' % (errno, klass.errno, klass.__name__) ) + + # Test format + if klass.format is not None: + assert klass.format is errors2.__messages[i] + i += 1 |