Source code for limits.limits

"""

"""
from six import add_metaclass

from functools import total_ordering


def safe_string(value):
    """
    consistently converts a value to a string
    :param value:
    :return: str
    """
    if isinstance(value, bytes):
        return value.decode()
    return str(value)


TIME_TYPES = dict(
    day=(60 * 60 * 24, "day"),
    month=(60 * 60 * 24 * 30, "month"),
    year=(60 * 60 * 24 * 30 * 12, "year"),
    hour=(60 * 60, "hour"),
    minute=(60, "minute"),
    second=(1, "second"),
)

GRANULARITIES = {}


class RateLimitItemMeta(type):
    def __new__(cls, name, parents, dct):
        granularity = super(RateLimitItemMeta, cls).__new__(cls, name, parents, dct)
        if "granularity" in dct:
            GRANULARITIES[dct["granularity"][1]] = granularity
        return granularity


# pylint: disable=no-member
[docs]@add_metaclass(RateLimitItemMeta) @total_ordering class RateLimitItem(object): """ defines a Rate limited resource which contains the characteristic namespace, amount and granularity multiples of the rate limiting window. :param int amount: the rate limit amount :param int multiples: multiple of the 'per' granularity (e.g. 'n' per 'm' seconds) :param string namespace: category for the specific rate limit """ __metaclass__ = RateLimitItemMeta __slots__ = ["namespace", "amount", "multiples", "granularity"] def __init__(self, amount, multiples=1, namespace="LIMITER"): self.namespace = namespace self.amount = int(amount) self.multiples = int(multiples or 1)
[docs] @classmethod def check_granularity_string(cls, granularity_string): """ checks if this instance matches a granularity string of type 'n per hour' etc. :return: True/False """ return granularity_string.lower() in cls.granularity[1:]
[docs] def get_expiry(self): """ :return: the size of the window in seconds. """ return self.granularity[0] * self.multiples
[docs] def key_for(self, *identifiers): """ :param identifiers: a list of strings to append to the key :return: a string key identifying this resource with each identifier appended with a '/' delimiter. """ remainder = "/".join( [safe_string(k) for k in identifiers] + [ safe_string(self.amount), safe_string(self.multiples), self.granularity[1], ] ) return "%s/%s" % (self.namespace, remainder)
def __eq__(self, other): return self.amount == other.amount and self.granularity == other.granularity def __repr__(self): return "%d per %d %s" % (self.amount, self.multiples, self.granularity[1]) def __lt__(self, other): return self.granularity[0] < other.granularity[0]
[docs]class RateLimitItemPerYear(RateLimitItem): """ per year rate limited resource. """ granularity = TIME_TYPES["year"]
[docs]class RateLimitItemPerMonth(RateLimitItem): """ per month rate limited resource. """ granularity = TIME_TYPES["month"]
[docs]class RateLimitItemPerDay(RateLimitItem): """ per day rate limited resource. """ granularity = TIME_TYPES["day"]
[docs]class RateLimitItemPerHour(RateLimitItem): """ per hour rate limited resource. """ granularity = TIME_TYPES["hour"]
[docs]class RateLimitItemPerMinute(RateLimitItem): """ per minute rate limited resource. """ granularity = TIME_TYPES["minute"]
[docs]class RateLimitItemPerSecond(RateLimitItem): """ per second rate limited resource. """ granularity = TIME_TYPES["second"]