blob: 2309fdc98023099d8f34c0d9e0f93d3e504cf311 [file] [log] [blame]
# Copyright 2014 The LUCI Authors. All rights reserved.
# Use of this source code is governed under the Apache License, Version 2.0
# that can be found in the LICENSE file.
"""Allows mockable access to the current time."""
from recipe_engine import recipe_api
import datetime
import functools
import time
import gevent
from recipe_engine.internal.global_shutdown import GLOBAL_SHUTDOWN
class exponential_retry:
"""Decorator which retries the function with exponential backoff.
See TimeApi.exponential_retry for full documentation.
"""
def __init__(self, retries, delay, condition=None, time_api=None):
"""Creates a new exponential retry decorator.
Args:
time_api (TimeApi): A TimeApi instance. Used to sleep.
retries (int): Maximum number of retries before giving up.
This value controls the number of *retries*, not the number of total
executions of the function.
If you decorate a function with a value of 3 retries, the function
will execute a maximum of 4 times; 1 time initially, and 3 more times
as it gets retried 3 times.
delay (datetime.timedelta): Amount of time to wait before retrying. This
will double every retry attempt (exponential).
This delay is 'jittered' to avoid the 'thundering herd' problem
(https://en.wikipedia.org/wiki/Thundering_herd_problem).
We only sleep in integral seconds, so sub-second resolution is not
supported for delays.
condition (func): If not None, a function that will be passed the
exception as its one argument. Retries will only happen if this
function returns True. If None, retries will always happen.
"""
# NOTE: Because the decorator can exist as module-level state, it is
# important that these values are READ-ONLY. Writing to them will act like
# assigning a global variable for that particular instance of the decorator,
# meaning that multiple different tests running in the same process could
# save arbitrary data which crosses between test cases.
self.time_api = time_api
self.retries = retries
self.delay = delay
self.condition = condition or (lambda e: True)
def __call__(self, f):
@functools.wraps(f)
def wrapper(*args, **kwargs):
time_api = self.time_api
if time_api is None:
try:
time_api = args[0].m.time
err_msg = "Could not find TimeAPI module. " \
"See docs for recipe_engine/time.exponential_retry for usage."
assert isinstance(time_api, TimeApi), err_msg
except:
raise
retry_delay = self.delay
# We want to retry self.retries times, so we make the range give us
# exactly self.retries loop executions.
for i in range(self.retries + 1):
try:
return f(*args, **kwargs)
except Exception as e:
if i >= self.retries or not self.condition(e):
raise
to_sleep = retry_delay.total_seconds()
# Jitter the amount to sleep by plus or minus 15%.
# Jitter helps avoid
# https://en.wikipedia.org/wiki/Thundering_herd_problem
to_sleep = time_api._jitter(to_sleep, .15)
time_api.sleep(to_sleep)
retry_delay *= 2
return wrapper
class TimeApi(recipe_api.RecipeApi):
def __init__(self, **kwargs):
super(TimeApi, self).__init__(**kwargs)
self._fake_time = None
self._fake_step = None
if self._test_data.enabled:
self._fake_time = self._test_data.get('seed', 1337000000.0)
self._fake_step = self._test_data.get('step', 1.5)
def sleep(self, secs, with_step=None, step_result=None):
"""Suspend execution of |secs| (float) seconds, waiting for GLOBAL_SHUTDOWN.
Does nothing in testing.
Args:
* secs (number) - The number of seconds to sleep.
* with_step (bool|None) - If True (or None and secs>60), emits a step to
indicate to users that the recipe is sleeping (not just hanging). False
suppresses this.
* step_result (step_data.StepData|None) - Result of running a step. Should
be None if with_step is True or None.
"""
if with_step is True or (
with_step is None and secs > 60): # pragma: no cover
assert step_result is None, (
'do not specify step_result if you want sleep to emit a new step')
step_result = self.m.step.empty('sleep %d' % (secs,))
if not self._test_data.enabled: # pragma: no cover
gevent.wait([GLOBAL_SHUTDOWN], timeout=secs)
if GLOBAL_SHUTDOWN.ready() and step_result:
step_result.presentation.status = "CANCELED"
def exponential_retry(self, retries, delay, condition=None):
"""Adds exponential retry to a function.
Decorator which retries the function with exponential backoff.
Each time the decorated function throws an exception, we sleep for some
amount of time. We increase the amount of time exponentially to prevent
cascading failures from overwhelming systems. We also add a jitter to avoid
the thundering herd problem.
Example usage:
```
def RunSteps(api):
@api.time.exponential_retry(5, datetime.timedelta(seconds=1))
def test_retries():
api.step('running', None)
raise Exception()
test_retries()
# Executes 6 steps with 'running' as a common prefix of their step names.
```
When writing a recipe module whose method needs to be retried, you won't
have access to the time module in the class body, but you can import a
class-method decorator like:
from RECIPE_MODULES.recipe_engine.time.api import exponential_retry
This decorator can be used on class methods or on functions
(for example, functions in a recipe file).
NOTE: Your module/recipe MUST ALSO depend on
"recipe_engine/time" in its DEPS.
NOTE: For non-class-method functions, the first parameter to those functions
must be an api object, such as the passed to RunSteps.
Example usage 1 (class method decorator):
```
from recipe_engine.recipe_api import RecipeApi
from RECIPE_MODULES.recipe_engine.time.api import exponential_retry
# NOTE: Don't forget to put "recipe_engine/time" in the module DEPS.
class MyRecipeModule(RecipeApi):
@exponential_retry(5, datetime.timedelta(seconds=1))
def my_retriable_function(self, ...):
self.m.step('running', None)
```
Example usage 2 (function with api as first arg):
```
from RECIPE_MODULES.recipe_engine.time.api import exponential_retry
# NOTE: Don't forget to put "recipe_engine/time" in DEPS.
@exponential_retry(5, datetime.timedelta(seconds=1))
def helper_function(api):
api.step('running', None)
def RunSteps(api):
helper_funciton(api)
```
"""
return exponential_retry(retries, delay, condition, self)
def time(self):
"""Returns current timestamp as a float number of seconds since epoch."""
if self._test_data.enabled:
self._fake_time += self._fake_step
return self._fake_time
else: # pragma: no cover
return time.time()
def ms_since_epoch(self):
"""Returns current timestamp as an int number of milliseconds since epoch.
"""
return int(round(self.time() * 1000))
def utcnow(self):
"""Returns current UTC time as a datetime.datetime."""
if self._test_data.enabled:
self._fake_time += self._fake_step
return datetime.datetime.utcfromtimestamp(self._fake_time)
else: # pragma: no cover
return datetime.datetime.utcnow()
def _jitter(self, seconds, jitter_amount, random_func=None):
"""Returns the provided seconds jittered by the jitter amount provided.
random_func allows for manually providing the random value in tests.
"""
if not random_func:
random_func = self.m.random.random
return seconds * (1 + random_func() * (jitter_amount * 2) - jitter_amount)
def timeout(self, seconds: float | int | datetime.timedelta = None):
"""Provides a context that times out after the given time.
Usage:
with api.time.timeout(datetime.timedelta(minutes=5)):
# your steps
Look at the "deadline" section of https://chromium.googlesource.com/infra/luci/luci-py/+/HEAD/client/LUCI_CONTEXT.md
to see how this works.
"""
if isinstance(seconds, datetime.timedelta):
seconds = seconds.total_seconds()
if seconds < 0:
raise recipe_api.StepFailure('`seconds` cannot be negative')
current_time = self.time()
# Make the deadline
deadline = self.m.context.deadline
deadline.soft_deadline = current_time + seconds
return self.m.context(deadline=deadline)