blob: a0b51fb25c372c443b425ecaf100021bf0380776 [file] [log] [blame]
# Copyright 2015 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Basic infrastructure for implementing retries.
This code is adopted from autotest: client/common_lib/cros/retry.py
This implementation removes the timeout feature as that requires the retry to
be done in main thread. For devserver, the call is handled in a thread kicked
off by cherrypy, so timeotu can't be supported.
"""
from __future__ import print_function
import cherrypy
import random
import sys
import time
def retry(ExceptionToCheck, timeout_min=1.0, delay_sec=3, blacklist=None):
"""Retry calling the decorated function using a delay with jitter.
Will raise RPC ValidationError exceptions from the decorated
function without retrying; a malformed RPC isn't going to
magically become good. Will raise exceptions in blacklist as well.
original from:
http://www.saltycrane.com/blog/2009/11/trying-out-retry-decorator-python/
Args:
ExceptionToCheck: the exception to check. May be a tuple of exceptions to
check.
timeout_min: timeout in minutes until giving up.
delay_sec: pre-jittered delay between retries in seconds. Actual delays
will be centered around this value, ranging up to 50% off this
midpoint.
blacklist: a list of exceptions that will be raised without retrying
"""
def deco_retry(func):
random.seed()
def delay():
"""'Jitter' the delay, up to 50% in either direction."""
random_delay = random.uniform(.5 * delay_sec, 1.5 * delay_sec)
cherrypy.log('Retrying in %f seconds...' % random_delay)
time.sleep(random_delay)
def func_retry(*args, **kwargs):
# Used to cache exception to be raised later.
exc_info = None
delayed_enabled = False
exception_tuple = () if blacklist is None else tuple(blacklist)
start_time = time.time()
remaining_time = timeout_min * 60
while remaining_time > 0:
if delayed_enabled:
delay()
else:
delayed_enabled = True
try:
# Clear the cache
exc_info = None
return func(*args, **kwargs)
except exception_tuple:
raise
except ExceptionToCheck as e:
cherrypy.log('%s(%s)' % (e.__class__, e))
# Cache the exception to be raised later.
exc_info = sys.exc_info()
remaining_time = int(timeout_min*60 - (time.time() - start_time))
# Raise the cached exception with original backtrace.
raise exc_info[0], exc_info[1], exc_info[2]
return func_retry # true decorator
return deco_retry