Last active
July 25, 2022 16:13
-
-
Save rsperl/9196b492f1b029ef24f15ec760776e84 to your computer and use it in GitHub Desktop.
decorator class, base class, and implementation class to provide customized retries for rest calls #python #rest #snippet
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from requests import Request, Session | |
import logging | |
import retry_decorator | |
retry_5x = retry_decorator.RetryRest(retries=5) | |
class RestBase(object): | |
def __init__(self, url, username, password): | |
self.url = url | |
self.username = username | |
self.password = password | |
self.session = Session() | |
self.log = logging.getLogger(__name__) | |
def do(self, method, url, data=dict(), params=dict()): | |
try: | |
resp = self._do(method, url, data, params) | |
return resp.status_code, resp.content | |
except Exception as e: | |
self.log.debug("exception calling do({}, {}): {}".format(method, url, e)) | |
@retry_5x | |
def _do(self, method, url, data=dict(), params=dict()): | |
method = method.lower() | |
headers = { | |
"accept": "application/json", | |
"content-type": "application/json", | |
} | |
r = None | |
url = self.url + url | |
self.log.info("{} {} {}".format(method, url, params)) | |
r = Request( | |
method=method, | |
url=url, | |
json=data, | |
params=params, | |
headers=headers, | |
auth=(self.username, self.password) | |
) | |
prepped = r.prepare() | |
return self.session.send(prepped, verify=False) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/env python | |
# -*- coding: utf-8 -*- | |
# src: https://powerfulpython.com/blog/making-unreliable-apis-reliable-with-python/ | |
# | |
# example usage: | |
# retry_on_auth_failure = RetryOnAuthFailure() | |
# retry_on_server_failure = RetryOnServerFailure(retries=5) | |
# | |
# @retry_on_server_failure | |
# @retry_on_auth_failure | |
# def myfunc(): | |
# ... | |
import logging | |
import time | |
class RetryRest: | |
def __init__(self, retries=5, delay=None, exponentialBackup=True): | |
""" | |
Create a RetryRest decorator that uses five retries with exponential backup by default. | |
A delay can be passed to use a standard number of seconds in between tries. If no delay | |
is given, and exponentialBackup is false, then a wait time of 1s is assumed. | |
""" | |
self.retries = retries | |
self.delay = delay | |
self.logger = logging.getLogger(__name__) | |
self.default_delay = 1 | |
def is_valid(self, resp): | |
"""by default, anything not 2xx is invalid""" | |
return resp.status_code >= 200 or resp.status_code <= 299 | |
def __call__(self, func): | |
def retried_func(*args, **kwargs): | |
tries = 1 | |
while True: | |
self.logger.debug("try {} of {}".format(tries, self.retries)) | |
resp = func(*args, **kwargs) | |
if self.is_valid(resp) or tries > self.retries: | |
break | |
tries += 1 | |
wait_time = self.delay | |
if self.delay is not None: | |
time.sleep(wait_time) | |
elif self.exponentialBackoff: | |
wait_time = 2 ** self.retries - 1 | |
else: | |
wait_time = self.default_delay | |
time.sleep(wait_time) | |
return resp | |
return retried_func | |
class RetryRestOnAuthFailure(RetryRest): | |
def is_valid(self, resp): | |
"""makes retry more specific -- this would allow a 500 error, but not a 400 or 401""" | |
return not (resp.status_code >= 400 and resp.status_code < 500) | |
class RetryRestOnServerFailure(RetryRest): | |
def is_valid(self, resp): | |
return resp.status_code >= 500 and resp.status_code <= 500 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
from rest_base import RestBase | |
class MyClient(RestBase): | |
def get_user(self, name): | |
url = "/users/" + name | |
return self.do("GET", url) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import sample_client | |
c = sample_client.MyClient(url="http://host", username="user", password="pass") | |
# .do will use the retry decorator from parent class to get a user | |
code, body = c.get_user("me") |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment