1
# Copyright 2011 OpenStack Foundation.
2
# Copyright 2012, Red Hat, Inc.
4
# Licensed under the Apache License, Version 2.0 (the "License"); you may
5
# not use this file except in compliance with the License. You may obtain
6
# a copy of the License at
8
# http://www.apache.org/licenses/LICENSE-2.0
10
# Unless required by applicable law or agreed to in writing, software
11
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13
# License for the specific language governing permissions and limitations
17
Exception related utilities.
27
from ironic.openstack.common.gettextutils import _LE
30
class save_and_reraise_exception(object):
31
"""Save current exception, run some code and then re-raise.
33
In some cases the exception context can be cleared, resulting in None
34
being attempted to be re-raised after an exception handler is run. This
35
can happen when eventlet switches greenthreads or when running an
36
exception handler, code raises and catches an exception. In both
37
cases the exception context will be cleared.
39
To work around this, we save the exception state, run handler code, and
40
then re-raise the original exception. If another exception occurs, the
41
saved exception is logged and the new exception is re-raised.
43
In some cases the caller may not want to re-raise the exception, and
44
for those circumstances this context provides a reraise flag that
45
can be used to suppress the exception. For example::
48
with save_and_reraise_exception() as ctxt:
49
decide_if_need_reraise()
50
if not should_be_reraised:
53
If another exception occurs and reraise flag is False,
54
the saved exception will not be logged.
56
If the caller wants to raise new exception during exception handling
57
he/she sets reraise to False initially with an ability to set it back to
61
with save_and_reraise_exception(reraise=False) as ctxt:
62
[if statements to determine whether to raise a new exception]
63
# Not raising a new exception, so reraise
66
def __init__(self, reraise=True):
67
self.reraise = reraise
70
self.type_, self.value, self.tb, = sys.exc_info()
73
def __exit__(self, exc_type, exc_val, exc_tb):
74
if exc_type is not None:
76
logging.error(_LE('Original exception being dropped: %s'),
77
traceback.format_exception(self.type_,
82
six.reraise(self.type_, self.value, self.tb)
85
def forever_retry_uncaught_exceptions(infunc):
86
def inner_func(*args, **kwargs):
88
last_exc_message = None
92
return infunc(*args, **kwargs)
93
except Exception as exc:
94
this_exc_message = six.u(str(exc))
95
if this_exc_message == last_exc_message:
99
# Do not log any more frequently than once a minute unless
100
# the exception message changes
101
cur_time = int(time.time())
102
if (cur_time - last_log_time > 60 or
103
this_exc_message != last_exc_message):
105
_LE('Unexpected exception occurred %d time(s)... '
106
'retrying.') % exc_count)
107
last_log_time = cur_time
108
last_exc_message = this_exc_message
110
# This should be a very rare event. In case it isn't, do